diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml
index 1252412..37c1b8e 100644
--- a/.github/workflows/test.yml
+++ b/.github/workflows/test.yml
@@ -9,12 +9,13 @@ jobs:
continue-on-error: ${{ matrix.continue-on-error }}
strategy:
matrix:
- python-version: ["3.8", "3.9", "3.10"]
+ python-version: ["3.8", "3.9", "3.10", "3.11"]
django: [32, 42]
- cms: [40]
+ cms: [40, 41]
requirements-file: [
dj32_cms40.txt,
dj42_cms40.txt,
+ dj42_cms41.txt,
]
continue-on-error: [true]
steps:
@@ -40,7 +41,7 @@ jobs:
- name: Install dependencies
run: |
python setup.py install
- pip install -r tests/requirements/dj${{matrix.django}}_cms${{matrix.cms}}.txt
+ pip install -r tests/requirements/${{matrix.requirements-file}}
sudo apt-get install gettext
python -m pip install --upgrade pip setuptools tox>4
- name: Test with tox
diff --git a/HISTORY.rst b/HISTORY.rst
index d254577..9bb7c55 100644
--- a/HISTORY.rst
+++ b/HISTORY.rst
@@ -10,6 +10,12 @@ History
2.1.0 (Unreleased)
==================
+Features
+--------
+
+- Add support for CMS 4.1.x
+
+
Features
--------
diff --git a/changes/41.feat b/changes/41.feat
new file mode 100644
index 0000000..0b0906d
--- /dev/null
+++ b/changes/41.feat
@@ -0,0 +1,2 @@
+* Add compatibility with DjangoCMS 4.1
+* Drop compatibility with DjangoCMS < 4.0
diff --git a/djangocms_page_sitemap/compat.py b/djangocms_page_sitemap/compat.py
new file mode 100644
index 0000000..ceb8b2a
--- /dev/null
+++ b/djangocms_page_sitemap/compat.py
@@ -0,0 +1,10 @@
+from cms import __version__ as CMS_VERSION
+from packaging.version import Version
+
+CMS_41 = Version("4.1") <= Version(CMS_VERSION)
+
+
+if CMS_41:
+ from cms.api import create_page_content # noqa: F401
+else:
+ from cms.api import create_title as create_page_content # noqa: F401
diff --git a/djangocms_page_sitemap/sitemap.py b/djangocms_page_sitemap/sitemap.py
index 676459b..4fe68aa 100644
--- a/djangocms_page_sitemap/sitemap.py
+++ b/djangocms_page_sitemap/sitemap.py
@@ -50,8 +50,8 @@ def items(self):
except ImportError:
return super().items().exclude(page__pagesitemapproperties__include_in_sitemap=False)
- def priority(self, title):
- ext_key = get_cache_key(title.page)
+ def priority(self, page_url):
+ ext_key = get_cache_key(page_url.page)
properties = cache.get(ext_key)
if properties:
return properties.priority
@@ -59,15 +59,15 @@ def priority(self, title):
try:
cache.set(
ext_key,
- title.page.pagesitemapproperties,
+ page_url.page.pagesitemapproperties,
PAGE_SITEMAP_CACHE_DURATION,
)
- return title.page.pagesitemapproperties.priority
+ return page_url.page.pagesitemapproperties.priority
except PageSitemapProperties.DoesNotExist:
return self.default_priority
- def changefreq(self, title):
- ext_key = get_cache_key(title.page)
+ def changefreq(self, page_url):
+ ext_key = get_cache_key(page_url.page)
properties = cache.get(ext_key)
if properties: # pragma: no cover
return properties.changefreq
@@ -75,10 +75,10 @@ def changefreq(self, title):
try:
cache.set(
ext_key,
- title.page.pagesitemapproperties,
+ page_url.page.pagesitemapproperties,
PAGE_SITEMAP_CACHE_DURATION,
)
- return title.page.pagesitemapproperties.changefreq
+ return page_url.page.pagesitemapproperties.changefreq
except PageSitemapProperties.DoesNotExist:
return self.default_changefreq
diff --git a/tests/base.py b/tests/base.py
index 4611dcf..3adbb24 100644
--- a/tests/base.py
+++ b/tests/base.py
@@ -1,11 +1,13 @@
from io import StringIO
-from cms.api import create_page, create_title
+from cms.api import create_page
+from cms.models import PageContent
from cms.utils.i18n import get_language_list
from django.contrib.auth.models import User
from django.http import HttpResponse, SimpleCookie
from django.test import RequestFactory, TestCase
+from djangocms_page_sitemap.compat import create_page_content
from djangocms_page_sitemap.utils import is_versioning_enabled
@@ -30,36 +32,6 @@ def setUpClass(cls):
cls.user_normal = User.objects.create(username="normal")
def get_pages(self):
- try:
- from cms.models import PageContent # noqa: F401
-
- return self._cms_4_pages()
- except ImportError:
- return self._cms_3_pages()
-
- def _cms_3_pages(self):
- page_1 = create_page("page one", "page.html", language="en")
- page_2 = create_page("page two", "page.html", language="en")
- page_3 = create_page("page three", "page.html", language="en")
- create_title(language="fr", title="page un", page=page_1)
- create_title(language="it", title="pagina uno", page=page_1)
- create_title(language="fr", title="page trois", page=page_3)
- for lang in self.languages:
- page_1.publish(lang)
- page_2.publish("en")
- page_3.publish("en")
- page_3.publish("fr")
- if hasattr(page_1, "set_as_homepage"):
- page_1.set_as_homepage()
-
- return (
- page_1.get_draft_object(),
- page_2.get_draft_object(),
- page_3.get_draft_object(),
- )
-
- def _cms_4_pages(self):
- from cms.models import PageContent # noqa: F401
page_1 = create_page("page one", "page.html", language="en", created_by=self.user)
page_2 = create_page("page two", "page.html", language="en", created_by=self.user)
@@ -67,9 +39,9 @@ def _cms_4_pages(self):
page_content1 = PageContent._base_manager.get(page=page_1, language="en")
page_content2 = PageContent._base_manager.get(page=page_2, language="en")
page_content3 = PageContent._base_manager.get(page=page_3, language="en")
- page_1_content_fr = create_title(language="fr", title="page un", page=page_1, created_by=self.user)
- page_1_content_it = create_title(language="it", title="pagina uno", page=page_1, created_by=self.user)
- page_3_content_fr = create_title(language="fr", title="page trois", page=page_3, created_by=self.user)
+ page_1_content_fr = create_page_content(language="fr", title="page un", page=page_1, created_by=self.user)
+ page_1_content_it = create_page_content(language="it", title="pagina uno", page=page_1, created_by=self.user)
+ page_3_content_fr = create_page_content(language="fr", title="page trois", page=page_3, created_by=self.user)
if is_versioning_enabled():
page_content1.versions.first().publish(self.user)
page_content2.versions.first().publish(self.user)
diff --git a/tests/requirements/dj32_cms40.txt b/tests/requirements/dj32_cms40.txt
index 80e163e..d60bfad 100644
--- a/tests/requirements/dj32_cms40.txt
+++ b/tests/requirements/dj32_cms40.txt
@@ -2,3 +2,4 @@
Django>=3.2,<4
+https://github.com/django-cms/django-cms/tarball/release/4.0.1.x#egg=django-cms
diff --git a/tests/requirements/dj42_cms40.txt b/tests/requirements/dj42_cms40.txt
index 9107cb2..64a91e0 100644
--- a/tests/requirements/dj42_cms40.txt
+++ b/tests/requirements/dj42_cms40.txt
@@ -2,3 +2,4 @@
Django>=4.2,<5
+https://github.com/django-cms/django-cms/tarball/release/4.0.1.x#egg=django-cms
diff --git a/tests/requirements/dj42_cms41.txt b/tests/requirements/dj42_cms41.txt
new file mode 100644
index 0000000..41ab8e9
--- /dev/null
+++ b/tests/requirements/dj42_cms41.txt
@@ -0,0 +1,5 @@
+-r requirements_base.txt
+
+Django>=4.2,<5
+
+https://github.com/django-cms/django-cms/tarball/release/4.1.x#egg=django-cms
diff --git a/tests/requirements/requirements_base.txt b/tests/requirements/requirements_base.txt
index 4ad6077..4f6ebe2 100644
--- a/tests/requirements/requirements_base.txt
+++ b/tests/requirements/requirements_base.txt
@@ -3,5 +3,3 @@ coverage>5
coveralls>2
mock>=1.0.1
django-app-helper>=2.0.0
-
-https://github.com/django-cms/django-cms/tarball/release/4.0.1.x#egg=django-cms
\ No newline at end of file
diff --git a/tests/test_sitemap.py b/tests/test_sitemap.py
index facd270..58f0bf8 100644
--- a/tests/test_sitemap.py
+++ b/tests/test_sitemap.py
@@ -2,11 +2,12 @@
from unittest import skipIf
import cms
-from cms.api import create_page, create_title
+from cms.api import create_page
from cms.test_utils.util.fuzzy_int import FuzzyInt
from django.core.cache import cache
from django.utils.timezone import now
+from djangocms_page_sitemap.compat import CMS_41, create_page_content
from djangocms_page_sitemap.models import PageSitemapProperties
from djangocms_page_sitemap.sitemap import ExtendedSitemap
from djangocms_page_sitemap.utils import get_cache_key, is_versioning_enabled
@@ -19,14 +20,14 @@ def test_sitemap_base(self):
page1, page2, page3 = self.get_pages()
try:
- title_language = page1.get_title_obj().language
+ page_content_language = page1.get_content_obj().language if CMS_41 else page1.get_title_obj().language
except AttributeError:
- title_language = page1.get_page_content_obj_attribute("language")
+ page_content_language = page1.get_page_content_obj_attribute("language")
sitemap = self.client.get("/sitemap.xml")
test_string = (
"http://example.com/%s/%s"
- "monthly0.5" % (title_language, now().strftime("%Y-%m-%d"))
+ "monthly0.5" % (page_content_language, now().strftime("%Y-%m-%d"))
)
self.assertContains(sitemap, test_string)
@@ -37,12 +38,12 @@ def test_sitemap_extended(self):
page1.publish("it")
try:
- title_language = page1.get_title_obj().language
+ page_content_language = page1.get_content_obj().language if CMS_41 else page1.get_title_obj().language
except AttributeError:
- title_language = page1.get_page_content_obj_attribute("language")
+ page_content_language = page1.get_page_content_obj_attribute("language")
test_string = (
"http://example.com/%s/%s"
- "never0.2" % (title_language, now().strftime("%Y-%m-%d"))
+ "never0.2" % (page_content_language, now().strftime("%Y-%m-%d"))
)
sitemap = self.client.get("/sitemap.xml")
self.assertContains(sitemap, test_string)
@@ -104,7 +105,7 @@ def test_pageurl_lastmod_with_cms4_versioning(self):
# Check the latest version modified date for the page is checked for lastmod()
# if versioning is enabled, Currenly test is skipped , as this require changes in testsuite
page_1 = create_page("page-one", "page.html", language="en", created_by=self.user)
- page_content = create_title(title="page un", language="en", page=page_1, created_by=self.user)
+ page_content = create_page_content(title="page un", language="en", page=page_1, created_by=self.user)
if is_versioning_enabled():
page_content.versions.first().publish(self.user)
last_modified_date = "%s" % (page_content.versions.first().modified.strftime("%Y-%m-%d"))
diff --git a/tests/test_toolbar.py b/tests/test_toolbar.py
index bd5ad92..e37743f 100644
--- a/tests/test_toolbar.py
+++ b/tests/test_toolbar.py
@@ -26,7 +26,8 @@ def find_toolbar_buttons(button_name, toolbar):
"""
found = []
for button_list in toolbar.get_right_items():
- found = found + [button for button in button_list.buttons if button.name == button_name]
+ if hasattr(button_list, "buttons"):
+ found = found + [button for button in button_list.buttons if button.name == button_name]
return found
diff --git a/tox.ini b/tox.ini
index 9d7592a..21b4ff0 100644
--- a/tox.ini
+++ b/tox.ini
@@ -8,18 +8,21 @@ envlist =
ruff
pypi-description
towncrier
- py{39, 310, 311}-django{32,42}-cms{40}
+ py{38, 39, 310, 311}-django{32,42}-cms{40}
+ py{38, 39, 310, 311}-django{42}-cms{41}
[testenv]
commands = {env:COMMAND:python} cms_helper.py djangocms_page_sitemap test {posargs}
deps =
-r{toxinidir}/tests/requirements/requirements_base.txt
- django32: -r{toxinidir}/tests/requirements/dj32_cms40.txt
- django42: -r{toxinidir}/tests/requirements/dj42_cms40.txt
+ django32_cms40: -r{toxinidir}/tests/requirements/dj32_cms40.txt
+ django42_cms40: -r{toxinidir}/tests/requirements/dj42_cms40.txt
+ django42_cms41: -r{toxinidir}/tests/requirements/dj42_cms41.txt
basepython=
py38: python3.8
py39: python3.9
py310: python3.10
+ py311: python3.11
passenv =
COMMAND
PYTEST_*