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.
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: commonground-api-common
3
- Version: 2.5.5
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.5.5.data/scripts/generate_schema,sha256=OpKgzlFc_uzA3TVW_vHSYXAD_feLaCdTEnkWjIcxVzA,280
2
- vng_api_common/__init__.py,sha256=UutRtbE_ZOD7CIHaQC80vf8Bv_vmHDWhwjlCbt3yseM,22
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=NdrZJqP7p54lRKoKG4mEDS9MqaMcPESIr4dB1tnLaqI,10163
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=VCXdU4q3CQ1cvuYkg1dPWSbZym1Ufoz5gSmbHYkQcgw,521
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.5.5.dist-info/METADATA,sha256=-ob8zNCm7QtO8IekDfwux_xZx00EdHYWqEKoqa6_sNU,6988
187
- commonground_api_common-2.5.5.dist-info/WHEEL,sha256=1tXe9gY0PYatrMPMDd6jXqjfpz_B-Wqm32CPfRC58XU,91
188
- commonground_api_common-2.5.5.dist-info/top_level.txt,sha256=vPismc83zPzWXTmlNCCwfDlFV9iygJYxNJW5iDjKTgw,15
189
- commonground_api_common-2.5.5.dist-info/RECORD,,
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,,
@@ -1,5 +1,5 @@
1
1
  Wheel-Version: 1.0
2
- Generator: setuptools (77.0.3)
2
+ Generator: setuptools (78.1.0)
3
3
  Root-Is-Purelib: true
4
4
  Tag: py3-none-any
5
5
 
@@ -1 +1 @@
1
- __version__ = "2.5.5"
1
+ __version__ = "2.6.1"
@@ -12,6 +12,7 @@ def generate_jwt(client_id, secret, user_id, user_representation):
12
12
  secret=secret,
13
13
  user_id=user_id,
14
14
  user_representation=user_representation,
15
+ jwt_valid_for=5 * 60,
15
16
  )
16
17
  )
17
18
  return f"Bearer {auth._token}"
@@ -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 isodate
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, 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, relativedelta
24
+ from relativedeltafield import format_relativedelta, parse_relativedelta
22
25
  except ImportError:
23
26
  format_relativedelta = None
24
- relativedelta = None
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, datetime.timedelta):
32
+ if isinstance(value, relativedelta):
30
33
  return value
31
34
  try:
32
- parsed = isodate.parse_duration(str(value))
33
- except isodate.ISO8601Error:
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 relativedelta and isinstance(value, relativedelta):
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
- >>> class VerlengingSerializer(GegevensGroepSerializer):
168
- ... class Meta:
169
- ... model = Zaak
170
- ... gegevensgroep = 'verlenging'
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
- class LengthHyperlinkedRelatedField(serializers.HyperlinkedRelatedField):
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