From a2a2f7b90b0877da5cda5b54118b22d440dae4df Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Mats=20Kr=C3=BCger=20Svensson?= Date: Fri, 17 Nov 2023 15:13:00 +0100 Subject: [PATCH 1/4] proposal on how to store more info in the openapi.json --- django_api_decorator/openapi.py | 27 ++++++++++++++++++++------- 1 file changed, 20 insertions(+), 7 deletions(-) diff --git a/django_api_decorator/openapi.py b/django_api_decorator/openapi.py index 9bfab99..a5ac6b3 100644 --- a/django_api_decorator/openapi.py +++ b/django_api_decorator/openapi.py @@ -19,20 +19,28 @@ def get_resolved_url_patterns( base_patterns: Sequence[URLResolver | URLPattern], -) -> list[tuple[URLPattern, str]]: +) -> list[tuple[URLPattern, str, str]]: """ Given a list of base URL patterns, this function digs down into the URL hierarchy and evaluates the full URLs of all django views that have a simple path (e.g. no regex). Returns a list of tuples with each RoutePattern and its full URL. """ + + def combine_path(existing:str, append: str | None) -> str: + if not append: + return existing + elif existing == "": + return append + else: + return existing + ":" + append unresolved_patterns: list[tuple[URLResolver | URLPattern, str]] = [ - (url_pattern, "/") for url_pattern in base_patterns + (url_pattern, "/", "") for url_pattern in base_patterns ] resolved_urls: list[tuple[URLPattern, str]] = [] while len(unresolved_patterns): - url_pattern, url_prefix = unresolved_patterns.pop() + url_pattern, url_prefix, reverse_path = unresolved_patterns.pop() if not isinstance(url_pattern.pattern, RoutePattern): logger.debug("Skipping URL that is not simple (e.g. regex or locale url)") @@ -44,10 +52,10 @@ def get_resolved_url_patterns( # If we are dealing with a URL Resolver we should dig further down. if isinstance(url_pattern, URLResolver): unresolved_patterns += [ - (child_pattern, url) for child_pattern in url_pattern.url_patterns + (child_pattern, url, combine_path(reverse_path, url_pattern.namespace)) for child_pattern in url_pattern.url_patterns ] else: - resolved_urls.append((url_pattern, url)) + resolved_urls.append((url_pattern, url, combine_path(reverse_path, url_pattern.name))) return resolved_urls @@ -89,7 +97,7 @@ def replacer(match: re.Match[str]) -> str: def paths_and_types_for_view( - *, view_name: str, callback: Callable[..., HttpResponse], resolved_url: str + *, view_name: str, callback: Callable[..., HttpResponse], resolved_url: str, reverse_path: str, ) -> tuple[dict[str, Any], dict[str, Any]]: api_meta: ApiMeta | None = getattr(callback, "_api_meta", None) @@ -162,6 +170,7 @@ def to_ref_if_object(schema: dict[str, Any]) -> dict[str, Any]: "description": textwrap.dedent(callback.__doc__ or "").strip(), # Tags are useful for grouping operations in codegen "tags": [app_name], + "x-reverse-path": reverse_path, "parameters": parameters, **request_body, "responses": { @@ -189,6 +198,7 @@ class OpenApiOperation: callback: Callable[..., HttpResponse] name: str url: str + reverse_path: str all_urls = get_resolved_url_patterns(urlpatterns) @@ -196,7 +206,7 @@ class OpenApiOperation: # Iterate through all django views within the url patterns and generate specs for # them - for pattern, resolved_url in all_urls: + for pattern, resolved_url, reverse_path in all_urls: if pattern.callback is None: continue @@ -226,6 +236,7 @@ class OpenApiOperation: f"{pattern.name or pattern.callback.__name__}" ), url=resolved_url, + reverse_path=reverse_path, ) ) @@ -235,6 +246,7 @@ class OpenApiOperation: callback=pattern.callback, name=pattern.name or pattern.callback.__name__, url=resolved_url, + reverse_path=reverse_path, ) ) @@ -251,6 +263,7 @@ class OpenApiOperation: view_name=operation.name, callback=operation.callback, resolved_url=operation.url, + reverse_path=operation.reverse_path, ) api_components.update(components) From 58063200ed24dbdfff24725776e39f048687999f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Mats=20Kr=C3=BCger=20Svensson?= Date: Mon, 20 Nov 2023 10:53:25 +0100 Subject: [PATCH 2/4] tests --- django_api_decorator/openapi.py | 21 ++++++++++++------- tests/test_openapi.py | 36 +++++++++++++++++++++++++++++---- tests/test_response_encoding.py | 7 +++++++ 3 files changed, 53 insertions(+), 11 deletions(-) diff --git a/django_api_decorator/openapi.py b/django_api_decorator/openapi.py index a5ac6b3..d99a58a 100644 --- a/django_api_decorator/openapi.py +++ b/django_api_decorator/openapi.py @@ -25,8 +25,8 @@ def get_resolved_url_patterns( and evaluates the full URLs of all django views that have a simple path (e.g. no regex). Returns a list of tuples with each RoutePattern and its full URL. """ - - def combine_path(existing:str, append: str | None) -> str: + + def combine_path(existing: str, append: str | None) -> str: if not append: return existing elif existing == "": @@ -34,10 +34,10 @@ def combine_path(existing:str, append: str | None) -> str: else: return existing + ":" + append - unresolved_patterns: list[tuple[URLResolver | URLPattern, str]] = [ + unresolved_patterns: list[tuple[URLResolver | URLPattern, str, str]] = [ (url_pattern, "/", "") for url_pattern in base_patterns ] - resolved_urls: list[tuple[URLPattern, str]] = [] + resolved_urls: list[tuple[URLPattern, str, str]] = [] while len(unresolved_patterns): url_pattern, url_prefix, reverse_path = unresolved_patterns.pop() @@ -52,10 +52,13 @@ def combine_path(existing:str, append: str | None) -> str: # If we are dealing with a URL Resolver we should dig further down. if isinstance(url_pattern, URLResolver): unresolved_patterns += [ - (child_pattern, url, combine_path(reverse_path, url_pattern.namespace)) for child_pattern in url_pattern.url_patterns + (child_pattern, url, combine_path(reverse_path, url_pattern.namespace)) + for child_pattern in url_pattern.url_patterns ] else: - resolved_urls.append((url_pattern, url, combine_path(reverse_path, url_pattern.name))) + resolved_urls.append( + (url_pattern, url, combine_path(reverse_path, url_pattern.name)) + ) return resolved_urls @@ -97,7 +100,11 @@ def replacer(match: re.Match[str]) -> str: def paths_and_types_for_view( - *, view_name: str, callback: Callable[..., HttpResponse], resolved_url: str, reverse_path: str, + *, + view_name: str, + callback: Callable[..., HttpResponse], + resolved_url: str, + reverse_path: str, ) -> tuple[dict[str, Any], dict[str, Any]]: api_meta: ApiMeta | None = getattr(callback, "_api_meta", None) diff --git a/tests/test_openapi.py b/tests/test_openapi.py index 8ad0543..be3fa45 100644 --- a/tests/test_openapi.py +++ b/tests/test_openapi.py @@ -4,11 +4,12 @@ from django.http import HttpRequest from django.test.client import Client from django.test.utils import override_settings -from django.urls import path +from django.urls import path, URLResolver, URLPattern +from django.urls.resolvers import RoutePattern from pydantic import BaseModel from django_api_decorator.decorators import api -from django_api_decorator.openapi import generate_api_spec +from django_api_decorator.openapi import generate_api_spec, get_resolved_url_patterns urlpatterns = None @@ -61,7 +62,7 @@ def view( return Response(state=State.OK, num=3) urls = [ - path("/", view), + path("/", view, name="view_name"), ] assert generate_api_spec(urls) == { @@ -70,9 +71,10 @@ def view( "paths": { "/{path_str}/{path_int}": { "post": { - "operationId": "view", + "operationId": "view_name", "description": "", "tags": ["test_openapi"], + "x-reverse-path": "view_name", "parameters": [ { "name": "path_str", @@ -239,6 +241,7 @@ def view(request: HttpRequest) -> A | B | C: "operationId": "view", "description": "", "tags": ["test_openapi"], + "x-reverse-path": "", "parameters": [], "responses": { 200: { @@ -282,3 +285,28 @@ def view(request: HttpRequest) -> A | B | C: } }, } + + +def test_get_resolved_url_patterns(): + child_pattern = URLPattern( + RoutePattern("child_pattern/nested/deep/"), lambda x: x, name="child_view" + ) + + base_patterns = [ + URLResolver( + pattern=RoutePattern("toplevel_pattern/"), + urlconf_name=[ + child_pattern, + ], + namespace="top_namespace", + ) + ] + + result = get_resolved_url_patterns(base_patterns) + assert result == [ + ( + child_pattern, + "/toplevel_pattern/child_pattern/nested/deep/", + "top_namespace:child_view", + ) + ] diff --git a/tests/test_response_encoding.py b/tests/test_response_encoding.py index ae1d5a1..841698e 100644 --- a/tests/test_response_encoding.py +++ b/tests/test_response_encoding.py @@ -113,6 +113,7 @@ def test_schema() -> None: "operationId": "view_union", "description": "", "tags": ["test_response_encoding"], + "x-reverse-path": "", "parameters": [], "responses": { 200: { @@ -136,6 +137,7 @@ def test_schema() -> None: "operationId": "view_camel_case_pydantic_model", "description": "", "tags": ["test_response_encoding"], + "x-reverse-path": "", "parameters": [], "responses": { 200: { @@ -156,6 +158,7 @@ def test_schema() -> None: "operationId": "view_pydantic_model", "description": "", "tags": ["test_response_encoding"], + "x-reverse-path": "", "parameters": [], "responses": { 200: { @@ -176,6 +179,7 @@ def test_schema() -> None: "operationId": "view_bool", "description": "", "tags": ["test_response_encoding"], + "x-reverse-path": "", "parameters": [], "responses": { 200: { @@ -192,6 +196,7 @@ def test_schema() -> None: "operationId": "view_int", "description": "", "tags": ["test_response_encoding"], + "x-reverse-path": "", "parameters": [], "responses": { 200: { @@ -208,6 +213,7 @@ def test_schema() -> None: "operationId": "view_typed_dict", "description": "", "tags": ["test_response_encoding"], + "x-reverse-path": "", "parameters": [], "responses": { 200: { @@ -228,6 +234,7 @@ def test_schema() -> None: "operationId": "view_json_response", "description": "", "tags": ["test_response_encoding"], + "x-reverse-path": "", "parameters": [], "responses": {200: {"description": ""}}, } From 7993fff111ccaf3d6d8d4dc7d25ed94eaa9adb15 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Mats=20Kr=C3=BCger=20Svensson?= Date: Mon, 20 Nov 2023 11:12:02 +0100 Subject: [PATCH 3/4] isort --- tests/test_openapi.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/test_openapi.py b/tests/test_openapi.py index be3fa45..fcb0851 100644 --- a/tests/test_openapi.py +++ b/tests/test_openapi.py @@ -4,7 +4,7 @@ from django.http import HttpRequest from django.test.client import Client from django.test.utils import override_settings -from django.urls import path, URLResolver, URLPattern +from django.urls import URLPattern, URLResolver, path from django.urls.resolvers import RoutePattern from pydantic import BaseModel From 2a16c5d9717b0977c5f382a1a78216239ab9fb82 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Mats=20Kr=C3=BCger=20Svensson?= Date: Mon, 20 Nov 2023 11:15:29 +0100 Subject: [PATCH 4/4] mypy --- tests/test_openapi.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/test_openapi.py b/tests/test_openapi.py index fcb0851..d0a3e74 100644 --- a/tests/test_openapi.py +++ b/tests/test_openapi.py @@ -287,7 +287,7 @@ def view(request: HttpRequest) -> A | B | C: } -def test_get_resolved_url_patterns(): +def test_get_resolved_url_patterns() -> None: child_pattern = URLPattern( RoutePattern("child_pattern/nested/deep/"), lambda x: x, name="child_view" )