commonground-api-common 2.5.5__py3-none-any.whl → 2.6.1__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- {commonground_api_common-2.5.5.dist-info → commonground_api_common-2.6.1.dist-info}/METADATA +1 -2
- {commonground_api_common-2.5.5.dist-info → commonground_api_common-2.6.1.dist-info}/RECORD +8 -8
- {commonground_api_common-2.5.5.dist-info → commonground_api_common-2.6.1.dist-info}/WHEEL +1 -1
- vng_api_common/__init__.py +1 -1
- vng_api_common/authorizations/utils.py +1 -0
- vng_api_common/serializers.py +126 -28
- {commonground_api_common-2.5.5.data → commonground_api_common-2.6.1.data}/scripts/generate_schema +0 -0
- {commonground_api_common-2.5.5.dist-info → commonground_api_common-2.6.1.dist-info}/top_level.txt +0 -0
{commonground_api_common-2.5.5.dist-info → commonground_api_common-2.6.1.dist-info}/METADATA
RENAMED
@@ -1,6 +1,6 @@
|
|
1
1
|
Metadata-Version: 2.4
|
2
2
|
Name: commonground-api-common
|
3
|
-
Version: 2.
|
3
|
+
Version: 2.6.1
|
4
4
|
Summary: Commonground API tooling
|
5
5
|
Home-page: https://github.com/maykinmedia/commonground-api-common
|
6
6
|
Author: Maykin Media, VNG-Realisatie
|
@@ -28,7 +28,6 @@ Requires-Dist: django-rest-framework-condition
|
|
28
28
|
Requires-Dist: drf-nested-routers>=0.94.1
|
29
29
|
Requires-Dist: drf-spectacular
|
30
30
|
Requires-Dist: iso-639
|
31
|
-
Requires-Dist: isodate
|
32
31
|
Requires-Dist: notifications-api-common>=0.3.1
|
33
32
|
Requires-Dist: zgw-consumers>=0.35.1
|
34
33
|
Requires-Dist: oyaml
|
@@ -1,5 +1,5 @@
|
|
1
|
-
commonground_api_common-2.
|
2
|
-
vng_api_common/__init__.py,sha256=
|
1
|
+
commonground_api_common-2.6.1.data/scripts/generate_schema,sha256=OpKgzlFc_uzA3TVW_vHSYXAD_feLaCdTEnkWjIcxVzA,280
|
2
|
+
vng_api_common/__init__.py,sha256=yv0wJuq7dd_PlBhLN8iuPUYVsoACKuk2R3Gg5WU-tHk,22
|
3
3
|
vng_api_common/admin.py,sha256=iFtUPGf-ha0I-bXgq8QIFrP23Kzk_H3FlgAjt0U-ip0,259
|
4
4
|
vng_api_common/apps.py,sha256=QQiJXRmjX9Q91oh0P9fvVnHe3NSYd1cEcUUBw0HLBCA,3690
|
5
5
|
vng_api_common/checks.py,sha256=tOyfV7MMLGh4anrd_W30LvJCxiyQ4sFs1mGd9mtrEc0,1175
|
@@ -28,7 +28,7 @@ vng_api_common/routers.py,sha256=hEnhBulkgMM-7W_lYaykKTgTBj3-avl7DGsR9P7BbTU,189
|
|
28
28
|
vng_api_common/schema.py,sha256=axs2Q8IXwpHNd5WscQg5xOErL6bWhP8WFItTt4xCFO4,16305
|
29
29
|
vng_api_common/scopes.py,sha256=PGs6CkXorAAdWXGFY1bSy-jmsPn122Njen9aFFOpFIQ,2351
|
30
30
|
vng_api_common/search.py,sha256=yehS6boCOk1JXLCqAMU-B62hWtbTBSf_WKIVGPgp0Mg,1045
|
31
|
-
vng_api_common/serializers.py,sha256=
|
31
|
+
vng_api_common/serializers.py,sha256=D0DEw-iw4se1MLAlDw25G6q4g1BTEL3VXcuvMTj6qlk,13462
|
32
32
|
vng_api_common/urls.py,sha256=9IWHYLlEIIHNaZ_Zq02qNQ2HJpETb7o-89r7yBM_tQs,270
|
33
33
|
vng_api_common/utils.py,sha256=EHqVjZhtqnbU7YrqgYIBss28Sd19jtnTLNaMWLfj3Zw,8203
|
34
34
|
vng_api_common/validators.py,sha256=ejaFZvFXFaBlqxjA2_07NSHKHlG5pejrfC_GHjwCj6E,12852
|
@@ -78,7 +78,7 @@ vng_api_common/authorizations/admin.py,sha256=Tk0yYKbb005E0XZaYYWbucMf_K5M8Hhz62
|
|
78
78
|
vng_api_common/authorizations/middleware.py,sha256=KJ3znCXPRMOVqSur62SmBjvC6RcKxtcWq1rzaHdYR98,8416
|
79
79
|
vng_api_common/authorizations/models.py,sha256=slIYxSktxCxSg03Nfb2mhsQse17b93KWE-rdPdMv8Ik,5199
|
80
80
|
vng_api_common/authorizations/serializers.py,sha256=3HeKWEqhI3UWwI8SttC4rEID-Epbk7SWsC-bEjolbaw,5151
|
81
|
-
vng_api_common/authorizations/utils.py,sha256=
|
81
|
+
vng_api_common/authorizations/utils.py,sha256=GmwTy5GhYk3e1VU4LpdfYdr8VD-R8p00hUY11QBbOhc,555
|
82
82
|
vng_api_common/authorizations/validators.py,sha256=u7fKm0QgGy8fiAeYmIEB9Gy-yIE9C-tC2ZpnNQBXPso,2816
|
83
83
|
vng_api_common/authorizations/migrations/0001_initial.py,sha256=ooAZtQeDtWgDxXzAP-KnSyyFYLRPM-PMrK5RgOnTPjQ,4360
|
84
84
|
vng_api_common/authorizations/migrations/0002_authorizationsconfig.py,sha256=m4taH6ClHI-YHYGGOKaq_qYXGx9lq1InXOGLQKg9MSw,1364
|
@@ -183,7 +183,7 @@ vng_api_common/tests/auth.py,sha256=IKDWTEFv4Bign4F70-ibsFcnJqRxEJaXvqaPQJWa1xY,
|
|
183
183
|
vng_api_common/tests/caching.py,sha256=zfIw5cRRvO9cekHZZKfRqZc8cx5IfJUYNmcH6cuIMg4,624
|
184
184
|
vng_api_common/tests/schema.py,sha256=WDvifDQQiKqIpQijpeQ7rYkFroJmuPuHe7zNhl1Bigk,2293
|
185
185
|
vng_api_common/tests/urls.py,sha256=PFrYzQbBC0TFPMEn3uPhcBG0IQs9JsEPqckicJT1UA4,2159
|
186
|
-
commonground_api_common-2.
|
187
|
-
commonground_api_common-2.
|
188
|
-
commonground_api_common-2.
|
189
|
-
commonground_api_common-2.
|
186
|
+
commonground_api_common-2.6.1.dist-info/METADATA,sha256=TCEnYBoRYR_1qTBTE73HroD-J4mDJEHuZOeROVkAcvo,6965
|
187
|
+
commonground_api_common-2.6.1.dist-info/WHEEL,sha256=CmyFI0kx5cdEMTLiONQRbGQwjIoR1aIYB7eCAQ4KPJ0,91
|
188
|
+
commonground_api_common-2.6.1.dist-info/top_level.txt,sha256=vPismc83zPzWXTmlNCCwfDlFV9iygJYxNJW5iDjKTgw,15
|
189
|
+
commonground_api_common-2.6.1.dist-info/RECORD,,
|
vng_api_common/__init__.py
CHANGED
@@ -1 +1 @@
|
|
1
|
-
__version__ = "2.
|
1
|
+
__version__ = "2.6.1"
|
vng_api_common/serializers.py
CHANGED
@@ -1,54 +1,45 @@
|
|
1
|
-
import datetime
|
2
1
|
import inspect
|
3
2
|
from collections import OrderedDict
|
3
|
+
from functools import reduce
|
4
4
|
from typing import List, Optional, Tuple, Union
|
5
5
|
|
6
6
|
from django.db import models, transaction
|
7
|
+
from django.db.models import Model
|
7
8
|
from django.utils.translation import gettext_lazy as _
|
8
9
|
|
9
|
-
import
|
10
|
+
from dateutil.relativedelta import relativedelta
|
10
11
|
from rest_framework import fields, serializers
|
12
|
+
from rest_framework.request import Request
|
13
|
+
from rest_framework_nested.relations import NestedHyperlinkedRelatedField
|
11
14
|
|
12
15
|
from .choices import TextChoicesWithDescriptions
|
13
16
|
from .descriptors import GegevensGroepType
|
14
17
|
|
15
18
|
try:
|
16
19
|
# 1.1.x
|
17
|
-
from relativedeltafield.utils import format_relativedelta,
|
20
|
+
from relativedeltafield.utils import format_relativedelta, parse_relativedelta
|
18
21
|
except ImportError:
|
19
22
|
try:
|
20
23
|
# 1.0.x
|
21
|
-
from relativedeltafield import format_relativedelta,
|
24
|
+
from relativedeltafield import format_relativedelta, parse_relativedelta
|
22
25
|
except ImportError:
|
23
26
|
format_relativedelta = None
|
24
|
-
|
27
|
+
parse_relativedelta = None
|
25
28
|
|
26
29
|
|
27
30
|
class DurationField(fields.DurationField):
|
28
31
|
def to_internal_value(self, value):
|
29
|
-
if isinstance(value,
|
32
|
+
if isinstance(value, relativedelta):
|
30
33
|
return value
|
31
34
|
try:
|
32
|
-
|
33
|
-
except
|
35
|
+
return parse_relativedelta(str(value))
|
36
|
+
except ValueError:
|
34
37
|
self.fail("invalid", format="P(n)Y(n)M(n)D")
|
35
|
-
else:
|
36
|
-
if isinstance(parsed, isodate.Duration):
|
37
|
-
# TODO: start should probably be a proper object, but we should
|
38
|
-
# really switch to relativedeltafield
|
39
|
-
parsed = parsed.totimedelta(start=datetime.datetime.now())
|
40
|
-
assert isinstance(parsed, datetime.timedelta)
|
41
|
-
return parsed
|
42
38
|
|
43
39
|
def to_representation(self, value) -> Optional[str]:
|
44
|
-
if
|
45
|
-
# relativedeltafield 1.1.2 returns a relativedelta() object with no duration,
|
46
|
-
# to keep behaviour consistent with older versions, change that to `None`
|
47
|
-
if not value:
|
48
|
-
return None
|
40
|
+
if isinstance(value, relativedelta):
|
49
41
|
return format_relativedelta(value)
|
50
|
-
|
51
|
-
return isodate.duration_isoformat(value)
|
42
|
+
return None
|
52
43
|
|
53
44
|
|
54
45
|
class FieldValidationErrorSerializer(serializers.Serializer):
|
@@ -164,11 +155,11 @@ class GegevensGroepSerializer(
|
|
164
155
|
|
165
156
|
Usage::
|
166
157
|
|
167
|
-
|
168
|
-
|
169
|
-
|
170
|
-
|
171
|
-
|
158
|
+
>>> class VerlengingSerializer(GegevensGroepSerializer):
|
159
|
+
... class Meta:
|
160
|
+
... model = Zaak
|
161
|
+
... gegevensgroep = 'verlenging'
|
162
|
+
>>>
|
172
163
|
|
173
164
|
Where ``Zaak.verlenging`` is a :class:``GegevensGroepType``.
|
174
165
|
"""
|
@@ -275,7 +266,77 @@ class NestedGegevensGroepMixin:
|
|
275
266
|
return super().update(instance, validated_data)
|
276
267
|
|
277
268
|
|
278
|
-
|
269
|
+
def get_nested_fk_attribute(instance, relation_path):
|
270
|
+
"""
|
271
|
+
Retrieves an attribute from a nested foreign key relation.
|
272
|
+
|
273
|
+
Args:
|
274
|
+
- instance: The model instance (e.g., Book).
|
275
|
+
- relation_path: A string with the relation path, e.g., 'author__publisher__name'.
|
276
|
+
|
277
|
+
Returns:
|
278
|
+
- The value of the nested attribute or None if not found.
|
279
|
+
"""
|
280
|
+
relations = relation_path.split("__")
|
281
|
+
|
282
|
+
return reduce(
|
283
|
+
lambda obj, attr: getattr(obj, attr, None) if obj else None, relations, instance
|
284
|
+
)
|
285
|
+
|
286
|
+
|
287
|
+
class CacheMixin:
|
288
|
+
"""
|
289
|
+
Mixin for Hyperlinked DRF fields to cache the base URI per view, to avoid
|
290
|
+
having to recalculate this for each related object that has to be serialized
|
291
|
+
|
292
|
+
This cache is stored on the field instance itself, so it's reset between requests
|
293
|
+
"""
|
294
|
+
|
295
|
+
lookup_url_kwarg = "" # Should be defined on `HyperlinkedRelatedField`
|
296
|
+
identifier_placeholder = "id-placeholder"
|
297
|
+
|
298
|
+
def get_extra_reverse_kwargs(self) -> dict[str, str]:
|
299
|
+
"""
|
300
|
+
Hook to inject extra kwargs to be passed to `reverse()`
|
301
|
+
"""
|
302
|
+
return {}
|
303
|
+
|
304
|
+
def __init__(self, *args, **kwargs):
|
305
|
+
super().__init__(*args, **kwargs)
|
306
|
+
|
307
|
+
self._reverse_cache = {}
|
308
|
+
|
309
|
+
def get_url(
|
310
|
+
self, obj: Model, view_name: str, request: Request, format: str | None
|
311
|
+
) -> str | None:
|
312
|
+
# Unsaved objects will not yet have a valid URL.
|
313
|
+
if hasattr(obj, "pk") and obj.pk in (None, ""):
|
314
|
+
return None
|
315
|
+
|
316
|
+
base_url = self._reverse_cache.get(view_name)
|
317
|
+
|
318
|
+
if base_url is None:
|
319
|
+
# If not cached, compute and cache it for this request cycle
|
320
|
+
try:
|
321
|
+
# Insert placeholders for identifiers, these will be replaced by
|
322
|
+
# real identifiers later on
|
323
|
+
kwargs = {self.lookup_url_kwarg: self.identifier_placeholder}
|
324
|
+
kwargs.update(self.get_extra_reverse_kwargs())
|
325
|
+
|
326
|
+
base_url = self.reverse(
|
327
|
+
view_name, kwargs=kwargs, request=request, format=format
|
328
|
+
)
|
329
|
+
self._reverse_cache[view_name] = base_url
|
330
|
+
except Exception as e:
|
331
|
+
raise ValueError(f"Could not resolve reverse for {view_name}: {e}")
|
332
|
+
|
333
|
+
url = base_url.replace(
|
334
|
+
self.identifier_placeholder, str(getattr(obj, self.lookup_url_kwarg))
|
335
|
+
)
|
336
|
+
return url
|
337
|
+
|
338
|
+
|
339
|
+
class LengthHyperlinkedRelatedField(CacheMixin, serializers.HyperlinkedRelatedField):
|
279
340
|
default_error_messages = {
|
280
341
|
"max_length": _("Ensure this field has no more than {max_length} characters."),
|
281
342
|
"min_length": _("Ensure this field has at least {min_length} characters."),
|
@@ -296,3 +357,40 @@ class LengthHyperlinkedRelatedField(serializers.HyperlinkedRelatedField):
|
|
296
357
|
self.fail("min_length", max_length=self.min_length, length=len(data))
|
297
358
|
|
298
359
|
return super().to_internal_value(data)
|
360
|
+
|
361
|
+
|
362
|
+
class CachedHyperlinkedRelatedField(CacheMixin, serializers.HyperlinkedRelatedField):
|
363
|
+
"""
|
364
|
+
Subclass of ``serializers.HyperlinkedRelatedField`` that applies caching in
|
365
|
+
``.get_url()`` to ``reverse()`` calls to improve serialization performance
|
366
|
+
"""
|
367
|
+
|
368
|
+
|
369
|
+
class CachedHyperlinkedIdentityField(CacheMixin, serializers.HyperlinkedIdentityField):
|
370
|
+
"""
|
371
|
+
Subclass of ``serializers.HyperlinkedIdentityField`` that applies caching in
|
372
|
+
``.get_url()`` to ``reverse()`` calls to improve serialization performance
|
373
|
+
"""
|
374
|
+
|
375
|
+
|
376
|
+
class CachedNestedHyperlinkedRelatedField(CacheMixin, NestedHyperlinkedRelatedField):
|
377
|
+
"""
|
378
|
+
Subclass of ``serializers.HyperlinkedIdentityField`` that applies caching in
|
379
|
+
``.get_url()`` to ``reverse()`` calls to improve serialization performance
|
380
|
+
"""
|
381
|
+
|
382
|
+
def get_extra_reverse_kwargs(self) -> dict[str, str]:
|
383
|
+
return self.parent_lookup_kwargs
|
384
|
+
|
385
|
+
def get_url(
|
386
|
+
self, obj: Model, view_name: str, request: Request, format: str | None
|
387
|
+
) -> str | None:
|
388
|
+
url = super().get_url(obj, view_name, request, format)
|
389
|
+
|
390
|
+
if not url:
|
391
|
+
return None
|
392
|
+
|
393
|
+
# Replace the placeholder from the cached base URI with the actual identifier
|
394
|
+
for k, v in self.parent_lookup_kwargs.items():
|
395
|
+
url = url.replace(v, str(get_nested_fk_attribute(obj, v)))
|
396
|
+
return url
|
{commonground_api_common-2.5.5.data → commonground_api_common-2.6.1.data}/scripts/generate_schema
RENAMED
File without changes
|
{commonground_api_common-2.5.5.dist-info → commonground_api_common-2.6.1.dist-info}/top_level.txt
RENAMED
File without changes
|