django-camomilla-cms 6.0.0b15__py2.py3-none-any.whl → 6.0.0b17__py2.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.
Files changed (72) hide show
  1. camomilla/__init__.py +1 -1
  2. camomilla/contrib/modeltranslation/hvad_migration.py +9 -9
  3. camomilla/dynamic_pages_urls.py +6 -2
  4. camomilla/managers/pages.py +93 -8
  5. camomilla/model_api.py +14 -7
  6. camomilla/models/media.py +1 -1
  7. camomilla/models/menu.py +10 -4
  8. camomilla/models/page.py +189 -127
  9. camomilla/openapi/schema.py +17 -8
  10. camomilla/redirects.py +10 -0
  11. camomilla/serializers/base/__init__.py +6 -4
  12. camomilla/serializers/fields/__init__.py +5 -17
  13. camomilla/serializers/fields/related.py +10 -4
  14. camomilla/serializers/mixins/__init__.py +23 -195
  15. camomilla/serializers/mixins/fields.py +20 -0
  16. camomilla/serializers/mixins/filter_fields.py +57 -0
  17. camomilla/serializers/mixins/json.py +34 -0
  18. camomilla/serializers/mixins/language.py +32 -0
  19. camomilla/serializers/mixins/nesting.py +35 -0
  20. camomilla/serializers/mixins/optimize.py +91 -0
  21. camomilla/serializers/mixins/ordering.py +34 -0
  22. camomilla/serializers/mixins/page.py +58 -0
  23. camomilla/{contrib/rest_framework/serializer.py → serializers/mixins/translation.py} +16 -56
  24. camomilla/serializers/utils.py +5 -3
  25. camomilla/serializers/validators.py +6 -2
  26. camomilla/settings.py +10 -2
  27. camomilla/storages/default.py +12 -0
  28. camomilla/storages/optimize.py +2 -2
  29. camomilla/storages/overwrite.py +2 -2
  30. camomilla/templates/defaults/parts/menu.html +1 -1
  31. camomilla/templatetags/menus.py +3 -0
  32. camomilla/theme/__init__.py +1 -1
  33. camomilla/theme/{admin.py → admin/__init__.py} +22 -20
  34. camomilla/theme/admin/pages.py +46 -0
  35. camomilla/theme/admin/translations.py +13 -0
  36. camomilla/theme/apps.py +1 -5
  37. camomilla/translation.py +7 -1
  38. camomilla/urls.py +2 -5
  39. camomilla/utils/query_parser.py +167 -0
  40. camomilla/utils/translation.py +47 -5
  41. camomilla/views/base/__init__.py +35 -5
  42. camomilla/views/medias.py +1 -1
  43. camomilla/views/menus.py +0 -2
  44. camomilla/views/mixins/__init__.py +17 -69
  45. camomilla/views/mixins/bulk_actions.py +22 -0
  46. camomilla/views/mixins/language.py +33 -0
  47. camomilla/views/mixins/optimize.py +18 -0
  48. camomilla/views/mixins/pagination.py +12 -18
  49. camomilla/views/mixins/permissions.py +6 -0
  50. camomilla/views/pages.py +12 -2
  51. {django_camomilla_cms-6.0.0b15.dist-info → django_camomilla_cms-6.0.0b17.dist-info}/METADATA +23 -16
  52. django_camomilla_cms-6.0.0b17.dist-info/RECORD +132 -0
  53. {django_camomilla_cms-6.0.0b15.dist-info → django_camomilla_cms-6.0.0b17.dist-info}/WHEEL +1 -1
  54. tests/fixtures/__init__.py +17 -0
  55. tests/test_api.py +2 -11
  56. tests/test_camomilla_filters.py +7 -13
  57. tests/test_media.py +113 -0
  58. tests/test_menu.py +97 -0
  59. tests/test_model_api.py +68 -0
  60. tests/test_model_api_permissions.py +39 -0
  61. tests/test_model_api_register.py +393 -0
  62. tests/test_pages.py +343 -0
  63. tests/test_query_parser.py +58 -0
  64. tests/test_templates_context.py +111 -0
  65. tests/test_utils.py +64 -64
  66. tests/utils/api.py +28 -0
  67. tests/utils/media.py +9 -0
  68. camomilla/serializers/fields/json.py +0 -49
  69. django_camomilla_cms-6.0.0b15.dist-info/RECORD +0 -105
  70. {django_camomilla_cms-6.0.0b15.dist-info → django_camomilla_cms-6.0.0b17.dist-info/licenses}/LICENSE +0 -0
  71. {django_camomilla_cms-6.0.0b15.dist-info → django_camomilla_cms-6.0.0b17.dist-info}/top_level.txt +0 -0
  72. {camomilla/contrib/rest_framework → tests/utils}/__init__.py +0 -0
@@ -1,195 +1,23 @@
1
- import django
2
- from django.conf import settings as django_settings
3
- from django.db.models.aggregates import Max
4
- from django.db.models.functions import Coalesce
5
- from django.utils import translation
6
- from rest_framework import serializers
7
- from rest_framework.utils import model_meta
8
-
9
- from camomilla.models import UrlNode
10
- from camomilla.fields import ORDERING_ACCEPTED_FIELDS
11
- from camomilla.serializers.fields.related import RelatedField
12
- from camomilla.serializers.utils import build_standard_model_serializer
13
- from camomilla.serializers.validators import UniquePermalinkValidator
14
- from camomilla.utils import dict_merge
15
- from camomilla import settings
16
-
17
- if django.VERSION >= (4, 0):
18
- from django.db.models import JSONField as DjangoJSONField
19
- else:
20
- from django.contrib.postgres.fields import JSONField as DjangoJSONField
21
-
22
- from typing import TYPE_CHECKING
23
-
24
- if TYPE_CHECKING:
25
- from camomilla.models.page import AbstractPage
26
-
27
-
28
- # TODO: decide what to do with LangInfoMixin mixin!
29
- class LangInfoMixin(metaclass=serializers.SerializerMetaclass):
30
- """
31
- This mixin adds a "lang_info" field to the serializer.
32
- This field contains information about the languages available in the site.
33
- """
34
- lang_info = serializers.SerializerMethodField("get_lang_info", read_only=True)
35
-
36
- def get_lang_info(self, obj, *args, **kwargs):
37
- languages = []
38
- for key, language in django_settings.LANGUAGES:
39
- languages.append({"id": key, "name": language})
40
- return {
41
- "default": django_settings.LANGUAGE_CODE,
42
- "active": translation.get_language(),
43
- "site_languages": languages,
44
- }
45
-
46
- def get_default_field_names(self, *args):
47
- field_names = super().get_default_field_names(*args)
48
- self.action = getattr(
49
- self, "action", self.context and self.context.get("action", "list")
50
- )
51
- if self.action != "retrieve":
52
- return [f for f in field_names if f != "lang_info"]
53
- return field_names
54
-
55
-
56
- class SetupEagerLoadingMixin:
57
- """
58
- This mixin allows to use the setup_eager_loading method to optimize the queries.
59
- """
60
- @staticmethod
61
- def setup_eager_loading(queryset):
62
- return queryset
63
-
64
-
65
- class OrderingMixin:
66
- """
67
- This mixin allows to set the default value of an ordering field to the max value + 1.
68
- """
69
-
70
- def get_max_order(self, order_field):
71
- return self.Meta.model.objects.aggregate(
72
- max_order=Coalesce(Max(order_field), 0)
73
- )["max_order"]
74
-
75
- def _get_ordering_field_name(self):
76
- try:
77
- field_name = self.Meta.model._meta.ordering[0]
78
- if field_name[0] == "-":
79
- field_name = field_name[1:]
80
- return field_name
81
- except (AttributeError, IndexError):
82
- return None
83
-
84
- def build_standard_field(self, field_name, model_field):
85
- field_class, field_kwargs = super().build_standard_field(
86
- field_name, model_field
87
- )
88
- if (
89
- isinstance(model_field, ORDERING_ACCEPTED_FIELDS)
90
- and field_name == self._get_ordering_field_name()
91
- ):
92
- field_kwargs["default"] = self.get_max_order(field_name) + 1
93
- return field_class, field_kwargs
94
-
95
-
96
- class JSONFieldPatchMixin:
97
- """
98
- This mixin allows to patch JSONField values during partial updates.
99
- This means that, if a JSONField is present in the request and the requsest uses PATCH method,
100
- the serializer will merge the new data with the old one.
101
- """
102
-
103
- def is_json_field(self, attr, value, info):
104
- return (
105
- attr in info.fields
106
- and isinstance(info.fields[attr], DjangoJSONField)
107
- and isinstance(value, dict)
108
- )
109
-
110
- def update(self, instance, validated_data):
111
- if self.partial:
112
- info = model_meta.get_field_info(instance)
113
- for attr, value in validated_data.items():
114
- if self.is_json_field(attr, value, info):
115
- validated_data[attr] = dict_merge(
116
- getattr(instance, attr, {}), value
117
- )
118
- return super().update(instance, validated_data)
119
-
120
-
121
- class NestMixin:
122
- """
123
- This mixin automatically creates nested serializers for relational fields.
124
- The depth of the nesting can be set using the "depth" attribute of the Meta class.
125
- If the depth is not set, the serializer will use the value coming from the settings.
126
-
127
- CAMOMILLA = { "API": {"NESTING_DEPTH": 10} }
128
- """
129
-
130
- def __init__(self, *args, **kwargs):
131
- self._depth = kwargs.pop("depth", None)
132
- return super().__init__(*args, **kwargs)
133
-
134
- def build_nested_field(self, field_name, relation_info, nested_depth):
135
- return self.build_relational_field(field_name, relation_info, nested_depth)
136
-
137
- def build_relational_field(
138
- self, field_name, relation_info, nested_depth=settings.API_NESTING_DEPTH + 1
139
- ):
140
- nested_depth = nested_depth if self._depth is None else self._depth
141
- field_class, field_kwargs = super().build_relational_field(
142
- field_name, relation_info
143
- )
144
- if (
145
- field_class is RelatedField and nested_depth > 1
146
- ): # stop recursion one step before the jump :P
147
- field_kwargs["serializer"] = build_standard_model_serializer(
148
- relation_info[1], nested_depth - 1
149
- )
150
- return field_class, field_kwargs
151
-
152
-
153
- class AbstractPageMixin(serializers.ModelSerializer):
154
- """
155
- This mixin is needed to serialize AbstractPage models.
156
- It provides permalink validation and some extra fields serialization.
157
-
158
- Use it as a base class for your serializer if you need to serialize custom AbstractPage models.
159
- """
160
-
161
- breadcrumbs = serializers.SerializerMethodField()
162
- routerlink = serializers.CharField(read_only=True)
163
- template_file = serializers.SerializerMethodField()
164
-
165
- def get_template_file(self, instance: "AbstractPage"):
166
- return instance.get_template_path()
167
-
168
- def get_breadcrumbs(self, instance: "AbstractPage"):
169
- return instance.breadcrumbs
170
-
171
- @property
172
- def translation_fields(self):
173
- return super().translation_fields + ["permalink"]
174
-
175
- def get_default_field_names(self, *args):
176
- from camomilla.contrib.rest_framework.serializer import RemoveTranslationsMixin
177
-
178
- if RemoveTranslationsMixin in self.__class__.__bases__: # noqa: E501
179
- return super().get_default_field_names(*args)
180
- return (
181
- [f for f in super().get_default_field_names(*args) if f != "url_node"]
182
- + UrlNode.LANG_PERMALINK_FIELDS
183
- + ["permalink"]
184
- )
185
-
186
- def build_field(self, field_name, info, model_class, nested_depth):
187
- if field_name in UrlNode.LANG_PERMALINK_FIELDS + ["permalink"]:
188
- return serializers.CharField, {
189
- "required": False,
190
- "allow_blank": True,
191
- }
192
- return super().build_field(field_name, info, model_class, nested_depth)
193
-
194
- def get_validators(self):
195
- return super().get_validators() + [UniquePermalinkValidator()]
1
+ from .fields import FieldsOverrideMixin
2
+ from .filter_fields import FilterFieldsMixin
3
+ from .json import JSONFieldPatchMixin
4
+ from .language import LangInfoMixin
5
+ from .nesting import NestMixin
6
+ from .optimize import SetupEagerLoadingMixin
7
+ from .ordering import OrderingMixin
8
+ from .page import AbstractPageMixin
9
+ from .translation import TranslationsMixin, RemoveTranslationsMixin
10
+
11
+
12
+ __all__ = [
13
+ "FieldsOverrideMixin",
14
+ "FilterFieldsMixin",
15
+ "JSONFieldPatchMixin",
16
+ "LangInfoMixin",
17
+ "NestMixin",
18
+ "SetupEagerLoadingMixin",
19
+ "OrderingMixin",
20
+ "AbstractPageMixin",
21
+ "TranslationsMixin",
22
+ "RemoveTranslationsMixin",
23
+ ]
@@ -0,0 +1,20 @@
1
+ from django.db import models
2
+ from rest_framework import serializers
3
+
4
+ from structured.fields import StructuredJSONField as ModelStructuredJSONField
5
+ from camomilla.serializers.fields import FileField, ImageField, RelatedField
6
+ from structured.contrib.restframework import StructuredJSONField
7
+
8
+
9
+ class FieldsOverrideMixin:
10
+ """
11
+ This mixin automatically overrides the fields of the serializer with camomilla's backed ones.
12
+ """
13
+
14
+ serializer_field_mapping = {
15
+ **serializers.ModelSerializer.serializer_field_mapping,
16
+ models.FileField: FileField,
17
+ models.ImageField: ImageField,
18
+ ModelStructuredJSONField: StructuredJSONField,
19
+ }
20
+ serializer_related_field = RelatedField
@@ -0,0 +1,57 @@
1
+ from rest_framework import serializers
2
+ from camomilla.serializers.fields.related import RelatedField
3
+ from collections import defaultdict
4
+
5
+
6
+ class FilterFieldsMixin(serializers.ModelSerializer):
7
+ """
8
+ Mixin to filter fields from a serializer, including handling nested fields.
9
+ """
10
+
11
+ def __init__(self, *args, **kwargs):
12
+ self.inherited_fields_filter = kwargs.pop("inherited_fields_filter", [])
13
+ return super().__init__(*args, **kwargs)
14
+
15
+ inherited_fields_filter = []
16
+
17
+ def get_default_field_names(self, *args):
18
+ field_names = super().get_default_field_names(*args)
19
+ request = self.context.get("request", None)
20
+
21
+ if request is not None and request.method == "GET":
22
+ fields = request.query_params.get("fields", "").split(",")
23
+ fields = [f for f in fields if f != ""]
24
+ if len(self.inherited_fields_filter) > 0:
25
+ fields = self.inherited_fields_filter
26
+
27
+ self.filtered_fields = set()
28
+ self.childs_fields = defaultdict(set)
29
+ for field in fields:
30
+ if "__" in field:
31
+ parent_field, child_field = field.split("__", 1)
32
+ if parent_field in field_names:
33
+ self.filtered_fields.add(parent_field)
34
+ self.childs_fields[parent_field].add(child_field)
35
+ else:
36
+ if field in field_names:
37
+ self.filtered_fields.add(field)
38
+
39
+ if len(self.filtered_fields) > 0:
40
+ return list(self.filtered_fields)
41
+ else:
42
+ return field_names
43
+
44
+ return field_names
45
+
46
+ def build_field(self, field_name, info, model_class, nested_depth):
47
+ field_class, field_kwargs = super().build_field(
48
+ field_name, info, model_class, nested_depth
49
+ )
50
+ inherited_fields_filter = (
51
+ self.childs_fields.get(field_name, [])
52
+ if hasattr(self, "childs_fields")
53
+ else []
54
+ )
55
+ if len(inherited_fields_filter) > 0 and issubclass(field_class, RelatedField):
56
+ field_kwargs["inherited_fields_filter"] = list(inherited_fields_filter)
57
+ return field_class, field_kwargs
@@ -0,0 +1,34 @@
1
+ import django
2
+ from rest_framework.utils import model_meta
3
+
4
+ from camomilla.utils import dict_merge
5
+
6
+ if django.VERSION >= (4, 0):
7
+ from django.db.models import JSONField as DjangoJSONField
8
+ else:
9
+ from django.contrib.postgres.fields import JSONField as DjangoJSONField
10
+
11
+
12
+ class JSONFieldPatchMixin:
13
+ """
14
+ This mixin allows to patch JSONField values during partial updates.
15
+ This means that, if a JSONField is present in the request and the requsest uses PATCH method,
16
+ the serializer will merge the new data with the old one.
17
+ """
18
+
19
+ def is_json_field(self, attr, value, info):
20
+ return (
21
+ attr in info.fields
22
+ and isinstance(info.fields[attr], DjangoJSONField)
23
+ and isinstance(value, dict)
24
+ )
25
+
26
+ def update(self, instance, validated_data):
27
+ if self.partial:
28
+ info = model_meta.get_field_info(instance)
29
+ for attr, value in validated_data.items():
30
+ if self.is_json_field(attr, value, info):
31
+ validated_data[attr] = dict_merge(
32
+ getattr(instance, attr, {}), value
33
+ )
34
+ return super().update(instance, validated_data)
@@ -0,0 +1,32 @@
1
+ from django.conf import settings as django_settings
2
+ from django.utils import translation
3
+ from rest_framework import serializers
4
+
5
+
6
+ # TODO: decide what to do with LangInfoMixin mixin!
7
+ class LangInfoMixin(metaclass=serializers.SerializerMetaclass):
8
+ """
9
+ This mixin adds a "lang_info" field to the serializer.
10
+ This field contains information about the languages available in the site.
11
+ """
12
+
13
+ lang_info = serializers.SerializerMethodField("get_lang_info", read_only=True)
14
+
15
+ def get_lang_info(self, obj, *args, **kwargs):
16
+ languages = []
17
+ for key, language in django_settings.LANGUAGES:
18
+ languages.append({"id": key, "name": language})
19
+ return {
20
+ "default": django_settings.LANGUAGE_CODE,
21
+ "active": translation.get_language(),
22
+ "site_languages": languages,
23
+ }
24
+
25
+ def get_default_field_names(self, *args):
26
+ field_names = super().get_default_field_names(*args)
27
+ self.action = getattr(
28
+ self, "action", self.context and self.context.get("action", "list")
29
+ )
30
+ if self.action != "retrieve":
31
+ return [f for f in field_names if f != "lang_info"]
32
+ return field_names
@@ -0,0 +1,35 @@
1
+ from camomilla.serializers.fields.related import RelatedField
2
+ from camomilla.serializers.utils import build_standard_model_serializer
3
+ from camomilla import settings
4
+
5
+
6
+ class NestMixin:
7
+ """
8
+ This mixin automatically creates nested serializers for relational fields.
9
+ The depth of the nesting can be set using the "depth" attribute of the Meta class.
10
+ If the depth is not set, the serializer will use the value coming from the settings.
11
+
12
+ CAMOMILLA = { "API": {"NESTING_DEPTH": 10} }
13
+ """
14
+
15
+ def __init__(self, *args, **kwargs):
16
+ self._depth = kwargs.pop("depth", None)
17
+ return super().__init__(*args, **kwargs)
18
+
19
+ def build_nested_field(self, field_name, relation_info, nested_depth):
20
+ return self.build_relational_field(field_name, relation_info, nested_depth)
21
+
22
+ def build_relational_field(
23
+ self, field_name, relation_info, nested_depth=settings.API_NESTING_DEPTH + 1
24
+ ):
25
+ nested_depth = nested_depth if self._depth is None else self._depth
26
+ field_class, field_kwargs = super().build_relational_field(
27
+ field_name, relation_info
28
+ )
29
+ if (
30
+ field_class is RelatedField and nested_depth > 1
31
+ ): # stop recursion one step before the jump :P
32
+ field_kwargs["serializer"] = build_standard_model_serializer(
33
+ relation_info[1], nested_depth - 1
34
+ )
35
+ return field_class, field_kwargs
@@ -0,0 +1,91 @@
1
+ from rest_framework.utils import model_meta
2
+
3
+
4
+ class Optimizations:
5
+ only = set()
6
+ select_related = set()
7
+ prefetch_related = set()
8
+
9
+ def __init__(self):
10
+ self.only = set()
11
+ self.select_related = set()
12
+ self.prefetch_related = set()
13
+
14
+ def __str__(self):
15
+ return f"Optimizations(only={self.only}, select_related={self.select_related}, prefetch_related={self.prefetch_related})"
16
+
17
+
18
+ def recursive_extract_optimizations(fields, info, max_depth=100) -> Optimizations:
19
+ optimizations = Optimizations()
20
+ if max_depth == 0:
21
+ return optimizations
22
+ for field in fields:
23
+ if "__" in field and field not in info.fields:
24
+ field_part_1, field_part_2 = field.split("__", 1)
25
+ if field_part_1 in info.relations:
26
+ field_info = info.relations[field_part_1]
27
+ nested_info = model_meta.get_field_info(field_info.related_model)
28
+ if field_info.to_many:
29
+ optimizations.prefetch_related.add(field_part_1)
30
+ optimizations.only.add(field_part_1)
31
+ else:
32
+ optimizations.select_related.add(field_part_1)
33
+ optimizations.only.add(field_part_1)
34
+ nested_optimizations = recursive_extract_optimizations(
35
+ [field_part_2], nested_info, max_depth - 1
36
+ )
37
+
38
+ for nested_field in nested_optimizations.only:
39
+ optimizations.only.add(f"{field_part_1}__{nested_field}")
40
+ for nested_field in nested_optimizations.select_related:
41
+ optimizations.select_related.add(f"{field_part_1}__{nested_field}")
42
+ for nested_field in nested_optimizations.prefetch_related:
43
+ optimizations.prefetch_related.add(
44
+ f"{field_part_1}__{nested_field}"
45
+ )
46
+ else:
47
+ if field in info.relations:
48
+ if info.relations[field].to_many:
49
+ optimizations.prefetch_related.add(field)
50
+ else:
51
+ optimizations.select_related.add(field)
52
+ optimizations.only.add(field)
53
+ elif field in info.fields:
54
+ optimizations.only.add(field)
55
+ return optimizations
56
+
57
+
58
+ class SetupEagerLoadingMixin:
59
+ """
60
+ This mixin allows to use the setup_eager_loading method to optimize the queries.
61
+ """
62
+
63
+ @classmethod
64
+ def optimize_qs(cls, queryset, context=None):
65
+ if hasattr(cls, "setup_eager_loading"):
66
+ queryset = cls.setup_eager_loading(queryset, context=context)
67
+ return cls.auto_optimize_queryset(queryset, context=context)
68
+
69
+ @classmethod
70
+ def auto_optimize_queryset(cls, queryset, context=None):
71
+ request = context.get("request", None)
72
+ if request and request.method == "GET":
73
+ serializer_fields = cls(context=context).fields.keys()
74
+ filtered_fields = set()
75
+ for field in request.query_params.get("fields", "").split(","):
76
+ filtered_fields.add(field)
77
+ if len(filtered_fields) == 0:
78
+ filtered_fields = serializer_fields
79
+ model = getattr(cls.Meta, "model", None)
80
+ if not model:
81
+ return queryset
82
+ optimizations = recursive_extract_optimizations(
83
+ filtered_fields, model_meta.get_field_info(model)
84
+ )
85
+ if len(optimizations.only) > 0:
86
+ queryset = queryset.only(*optimizations.only)
87
+ if len(optimizations.select_related) > 0:
88
+ queryset = queryset.select_related(*optimizations.select_related)
89
+ if len(optimizations.prefetch_related) > 0:
90
+ queryset = queryset.prefetch_related(*optimizations.prefetch_related)
91
+ return queryset
@@ -0,0 +1,34 @@
1
+ from django.db.models.aggregates import Max
2
+ from django.db.models.functions import Coalesce
3
+ from camomilla.fields import ORDERING_ACCEPTED_FIELDS
4
+
5
+
6
+ class OrderingMixin:
7
+ """
8
+ This mixin allows to set the default value of an ordering field to the max value + 1.
9
+ """
10
+
11
+ def get_max_order(self, order_field):
12
+ return self.Meta.model.objects.aggregate(
13
+ max_order=Coalesce(Max(order_field), 0)
14
+ )["max_order"]
15
+
16
+ def _get_ordering_field_name(self):
17
+ try:
18
+ field_name = self.Meta.model._meta.ordering[0]
19
+ if field_name[0] == "-":
20
+ field_name = field_name[1:]
21
+ return field_name
22
+ except (AttributeError, IndexError):
23
+ return None
24
+
25
+ def build_standard_field(self, field_name, model_field):
26
+ field_class, field_kwargs = super().build_standard_field(
27
+ field_name, model_field
28
+ )
29
+ if (
30
+ isinstance(model_field, ORDERING_ACCEPTED_FIELDS)
31
+ and field_name == self._get_ordering_field_name()
32
+ ):
33
+ field_kwargs["default"] = self.get_max_order(field_name) + 1
34
+ return field_class, field_kwargs
@@ -0,0 +1,58 @@
1
+ from rest_framework import serializers
2
+ from camomilla.models import UrlNode
3
+ from camomilla.serializers.validators import UniquePermalinkValidator
4
+ from typing import TYPE_CHECKING
5
+
6
+ if TYPE_CHECKING:
7
+ from camomilla.models.page import AbstractPage
8
+
9
+
10
+ class AbstractPageMixin(serializers.ModelSerializer):
11
+ """
12
+ This mixin is needed to serialize AbstractPage models.
13
+ It provides permalink validation and some extra fields serialization.
14
+
15
+ Use it as a base class for your serializer if you need to serialize custom AbstractPage models.
16
+ """
17
+
18
+ breadcrumbs = serializers.SerializerMethodField()
19
+ routerlink = serializers.CharField(read_only=True)
20
+ template_file = serializers.SerializerMethodField()
21
+
22
+ def get_template_file(self, instance: "AbstractPage"):
23
+ return instance.get_template_path()
24
+
25
+ def get_breadcrumbs(self, instance: "AbstractPage"):
26
+ return instance.breadcrumbs
27
+
28
+ @property
29
+ def translation_fields(self):
30
+ return super().translation_fields + ["permalink"]
31
+
32
+ def get_default_field_names(self, *args):
33
+ from camomilla.serializers.mixins.translation import RemoveTranslationsMixin
34
+
35
+ default_fields = super().get_default_field_names(*args)
36
+ filtered_fields = getattr(self, "filtered_fields", [])
37
+ if len(filtered_fields) > 0:
38
+ return filtered_fields
39
+ if RemoveTranslationsMixin in self.__class__.__bases__: # noqa: E501
40
+ return default_fields
41
+ return list(
42
+ set(
43
+ [f for f in default_fields if f != "url_node"]
44
+ + UrlNode.LANG_PERMALINK_FIELDS
45
+ + ["permalink"]
46
+ )
47
+ )
48
+
49
+ def build_field(self, field_name, info, model_class, nested_depth):
50
+ if field_name in UrlNode.LANG_PERMALINK_FIELDS + ["permalink"]:
51
+ return serializers.CharField, {
52
+ "required": False,
53
+ "allow_blank": True,
54
+ }
55
+ return super().build_field(field_name, info, model_class, nested_depth)
56
+
57
+ def get_validators(self):
58
+ return super().get_validators() + [UniquePermalinkValidator()]