diff --git a/.readthedocs.yml b/.readthedocs.yml index 82473625..c9b38b4e 100644 --- a/.readthedocs.yml +++ b/.readthedocs.yml @@ -5,10 +5,10 @@ build: tools: python: "3.11" jobs: - pre_build: - - pip install poetry - - poetry config virtualenvs.create false - - poetry install --only docs + post_create_environment: + - python -m pip install poetry + post_install: + - VIRTUAL_ENV=$READTHEDOCS_VIRTUALENV_PATH poetry install --with docs sphinx: configuration: docs/source/conf.py diff --git a/poetry.lock b/poetry.lock index 10c20b09..ae6e3bb0 100644 --- a/poetry.lock +++ b/poetry.lock @@ -564,6 +564,24 @@ ssh = ["bcrypt (>=3.1.5)"] test = ["certifi", "cryptography-vectors (==43.0.3)", "pretend", "pytest (>=6.2.0)", "pytest-benchmark", "pytest-cov", "pytest-xdist"] test-randomorder = ["pytest-randomly"] +[[package]] +name = "deepdiff" +version = "8.0.1" +description = "Deep Difference and Search of any Python object/data. Recreate objects by adding adding deltas to each other." +optional = false +python-versions = ">=3.8" +files = [ + {file = "deepdiff-8.0.1-py3-none-any.whl", hash = "sha256:42e99004ce603f9a53934c634a57b04ad5900e0d8ed0abb15e635767489cbc05"}, + {file = "deepdiff-8.0.1.tar.gz", hash = "sha256:245599a4586ab59bb599ca3517a9c42f3318ff600ded5e80a3432693c8ec3c4b"}, +] + +[package.dependencies] +orderly-set = "5.2.2" + +[package.extras] +cli = ["click (==8.1.7)", "pyyaml (==6.0.1)"] +optimize = ["orjson"] + [[package]] name = "defusedxml" version = "0.7.1" @@ -899,6 +917,17 @@ rsa = ["cryptography (>=3.0.0)"] signals = ["blinker (>=1.4.0)"] signedtoken = ["cryptography (>=3.0.0)", "pyjwt (>=2.0.0,<3)"] +[[package]] +name = "orderly-set" +version = "5.2.2" +description = "Orderly set" +optional = false +python-versions = ">=3.8" +files = [ + {file = "orderly_set-5.2.2-py3-none-any.whl", hash = "sha256:f7a37c95a38c01cdfe41c3ffb62925a318a2286ea0a41790c057fc802aec54da"}, + {file = "orderly_set-5.2.2.tar.gz", hash = "sha256:52a18b86aaf3f5d5a498bbdb27bf3253a4e5c57ab38e5b7a56fa00115cd28448"}, +] + [[package]] name = "packaging" version = "24.2" @@ -1713,4 +1742,4 @@ webauthn = ["webauthn"] [metadata] lock-version = "2.0" python-versions = "^3.8" -content-hash = "94889a528b87acd6b918ae7cdcce3d9a677b43a6509cc109cf5b7f314d68f6a6" +content-hash = "c0aeebaf0e9835af17384f711136cf65bbbc05fc7f054b9ba631372b1e0531c6" diff --git a/pyproject.toml b/pyproject.toml index ccb98cae..914dc47f 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -57,6 +57,7 @@ pytest-django = "^4.5.2" tox = "^4.4.8" babel = "^2.12.1" pytest-mock = "^3.14.0" +deepdiff = "^8.0.1" [tool.poetry.group.code-quality.dependencies] black = "^23.1.0" diff --git a/testproject/testapp/tests/test_urls/__init__.py b/testproject/testapp/tests/test_urls/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/testproject/testapp/tests/test_urls/test_urls.py b/testproject/testapp/tests/test_urls/test_urls.py new file mode 100644 index 00000000..ac37c40d --- /dev/null +++ b/testproject/testapp/tests/test_urls/test_urls.py @@ -0,0 +1,52 @@ +import json +import pathlib + +import pytest +from deepdiff import DeepDiff +from django.urls import get_resolver + + +@pytest.mark.django_db +def test_urls_have_not_changed(settings): + BASE_DIR = settings.BASE_DIR + if isinstance(BASE_DIR, str): + BASE_DIR = pathlib.Path(BASE_DIR) + TEST_PATH = BASE_DIR / "testapp" / "tests" / "test_urls" + FILE_PATH = TEST_PATH / "urls_snapshot.json" + url_patterns = get_resolver().url_patterns + + def get_all_urls(patterns, prefix=""): + urls = [] + for pattern in patterns: + if hasattr(pattern, "url_patterns"): + urls += get_all_urls( + pattern.url_patterns, + prefix + pattern.pattern.regex.pattern, + ) + else: + pattern_str = prefix + pattern.pattern.regex.pattern + name = pattern.name if pattern.name else None + urls.append({"pattern": pattern_str, "name": name}) + return urls + + current_urls = sorted(get_all_urls(url_patterns), key=lambda x: x["pattern"]) + # api-root generates different regex pattern locally vs in CI + current_urls = [el for el in current_urls if el["name"] != "api-root"] + + if not FILE_PATH.exists(): + with open(FILE_PATH, "w") as f: + json.dump(current_urls, f, indent=2) + pytest.fail( + "URL snapshot not found. Created snapshot with current URLs. Re-run the test." # noqa: E501 + ) + + with open(FILE_PATH) as f: + saved_urls = json.load(f) + + diff = DeepDiff(current_urls, saved_urls) + if diff: + with open(FILE_PATH, "w") as f: + json.dump(current_urls, f, indent=2) + pytest.fail( + f"URL structure has changed. Updated snapshot with new URLs and names. Diff:\n\n{diff}" # noqa: E501 + ) diff --git a/testproject/testapp/tests/test_urls/urls_snapshot.json b/testproject/testapp/tests/test_urls/urls_snapshot.json new file mode 100644 index 00000000..659e01a3 --- /dev/null +++ b/testproject/testapp/tests/test_urls/urls_snapshot.json @@ -0,0 +1,134 @@ +[ + { + "pattern": "^auth/^jwt/create/?", + "name": "jwt-create" + }, + { + "pattern": "^auth/^jwt/refresh/?", + "name": "jwt-refresh" + }, + { + "pattern": "^auth/^jwt/verify/?", + "name": "jwt-verify" + }, + { + "pattern": "^auth/^o/(?P\\S+)/$", + "name": "provider-auth" + }, + { + "pattern": "^auth/^token/login/?$", + "name": "login" + }, + { + "pattern": "^auth/^token/logout/?$", + "name": "logout" + }, + { + "pattern": "^auth/^users/$", + "name": "user-list" + }, + { + "pattern": "^auth/^users/(?P[^/.]+)/$", + "name": "user-detail" + }, + { + "pattern": "^auth/^users/(?P[^/.]+)\\.(?P[a-z0-9]+)/?$", + "name": "user-detail" + }, + { + "pattern": "^auth/^users/activation/$", + "name": "user-activation" + }, + { + "pattern": "^auth/^users/activation\\.(?P[a-z0-9]+)/?$", + "name": "user-activation" + }, + { + "pattern": "^auth/^users/me/$", + "name": "user-me" + }, + { + "pattern": "^auth/^users/me\\.(?P[a-z0-9]+)/?$", + "name": "user-me" + }, + { + "pattern": "^auth/^users/resend_activation/$", + "name": "user-resend-activation" + }, + { + "pattern": "^auth/^users/resend_activation\\.(?P[a-z0-9]+)/?$", + "name": "user-resend-activation" + }, + { + "pattern": "^auth/^users/reset_password/$", + "name": "user-reset-password" + }, + { + "pattern": "^auth/^users/reset_password\\.(?P[a-z0-9]+)/?$", + "name": "user-reset-password" + }, + { + "pattern": "^auth/^users/reset_password_confirm/$", + "name": "user-reset-password-confirm" + }, + { + "pattern": "^auth/^users/reset_password_confirm\\.(?P[a-z0-9]+)/?$", + "name": "user-reset-password-confirm" + }, + { + "pattern": "^auth/^users/reset_username/$", + "name": "user-reset-username" + }, + { + "pattern": "^auth/^users/reset_username\\.(?P[a-z0-9]+)/?$", + "name": "user-reset-username" + }, + { + "pattern": "^auth/^users/reset_username_confirm/$", + "name": "user-reset-username-confirm" + }, + { + "pattern": "^auth/^users/reset_username_confirm\\.(?P[a-z0-9]+)/?$", + "name": "user-reset-username-confirm" + }, + { + "pattern": "^auth/^users/set_password/$", + "name": "user-set-password" + }, + { + "pattern": "^auth/^users/set_password\\.(?P[a-z0-9]+)/?$", + "name": "user-set-password" + }, + { + "pattern": "^auth/^users/set_username/$", + "name": "user-set-username" + }, + { + "pattern": "^auth/^users/set_username\\.(?P[a-z0-9]+)/?$", + "name": "user-set-username" + }, + { + "pattern": "^auth/^users\\.(?P[a-z0-9]+)/?$", + "name": "user-list" + }, + { + "pattern": "^webauthn-example/$", + "name": null + }, + { + "pattern": "^webauthn/^login/$", + "name": "webauthn_login" + }, + { + "pattern": "^webauthn/^login_request/$", + "name": "webauthn_login_request" + }, + { + "pattern": "^webauthn/^signup/(?P.+)/$", + "name": "webauthn_signup" + }, + { + "pattern": "^webauthn/^signup_request/$", + "name": "webauthn_signup_request" + } +]