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.
- camomilla/__init__.py +1 -1
- camomilla/contrib/modeltranslation/hvad_migration.py +9 -9
- camomilla/dynamic_pages_urls.py +6 -2
- camomilla/managers/pages.py +93 -8
- camomilla/model_api.py +14 -7
- camomilla/models/media.py +1 -1
- camomilla/models/menu.py +10 -4
- camomilla/models/page.py +189 -127
- camomilla/openapi/schema.py +17 -8
- camomilla/redirects.py +10 -0
- camomilla/serializers/base/__init__.py +6 -4
- camomilla/serializers/fields/__init__.py +5 -17
- camomilla/serializers/fields/related.py +10 -4
- camomilla/serializers/mixins/__init__.py +23 -195
- camomilla/serializers/mixins/fields.py +20 -0
- camomilla/serializers/mixins/filter_fields.py +57 -0
- camomilla/serializers/mixins/json.py +34 -0
- camomilla/serializers/mixins/language.py +32 -0
- camomilla/serializers/mixins/nesting.py +35 -0
- camomilla/serializers/mixins/optimize.py +91 -0
- camomilla/serializers/mixins/ordering.py +34 -0
- camomilla/serializers/mixins/page.py +58 -0
- camomilla/{contrib/rest_framework/serializer.py → serializers/mixins/translation.py} +16 -56
- camomilla/serializers/utils.py +5 -3
- camomilla/serializers/validators.py +6 -2
- camomilla/settings.py +10 -2
- camomilla/storages/default.py +12 -0
- camomilla/storages/optimize.py +2 -2
- camomilla/storages/overwrite.py +2 -2
- camomilla/templates/defaults/parts/menu.html +1 -1
- camomilla/templatetags/menus.py +3 -0
- camomilla/theme/__init__.py +1 -1
- camomilla/theme/{admin.py → admin/__init__.py} +22 -20
- camomilla/theme/admin/pages.py +46 -0
- camomilla/theme/admin/translations.py +13 -0
- camomilla/theme/apps.py +1 -5
- camomilla/translation.py +7 -1
- camomilla/urls.py +2 -5
- camomilla/utils/query_parser.py +167 -0
- camomilla/utils/translation.py +47 -5
- camomilla/views/base/__init__.py +35 -5
- camomilla/views/medias.py +1 -1
- camomilla/views/menus.py +0 -2
- camomilla/views/mixins/__init__.py +17 -69
- camomilla/views/mixins/bulk_actions.py +22 -0
- camomilla/views/mixins/language.py +33 -0
- camomilla/views/mixins/optimize.py +18 -0
- camomilla/views/mixins/pagination.py +12 -18
- camomilla/views/mixins/permissions.py +6 -0
- camomilla/views/pages.py +12 -2
- {django_camomilla_cms-6.0.0b15.dist-info → django_camomilla_cms-6.0.0b17.dist-info}/METADATA +23 -16
- django_camomilla_cms-6.0.0b17.dist-info/RECORD +132 -0
- {django_camomilla_cms-6.0.0b15.dist-info → django_camomilla_cms-6.0.0b17.dist-info}/WHEEL +1 -1
- tests/fixtures/__init__.py +17 -0
- tests/test_api.py +2 -11
- tests/test_camomilla_filters.py +7 -13
- tests/test_media.py +113 -0
- tests/test_menu.py +97 -0
- tests/test_model_api.py +68 -0
- tests/test_model_api_permissions.py +39 -0
- tests/test_model_api_register.py +393 -0
- tests/test_pages.py +343 -0
- tests/test_query_parser.py +58 -0
- tests/test_templates_context.py +111 -0
- tests/test_utils.py +64 -64
- tests/utils/api.py +28 -0
- tests/utils/media.py +9 -0
- camomilla/serializers/fields/json.py +0 -49
- django_camomilla_cms-6.0.0b15.dist-info/RECORD +0 -105
- {django_camomilla_cms-6.0.0b15.dist-info → django_camomilla_cms-6.0.0b17.dist-info/licenses}/LICENSE +0 -0
- {django_camomilla_cms-6.0.0b15.dist-info → django_camomilla_cms-6.0.0b17.dist-info}/top_level.txt +0 -0
- {camomilla/contrib/rest_framework → tests/utils}/__init__.py +0 -0
@@ -1,195 +1,23 @@
|
|
1
|
-
import
|
2
|
-
from
|
3
|
-
from
|
4
|
-
from
|
5
|
-
from
|
6
|
-
from
|
7
|
-
from
|
8
|
-
|
9
|
-
from
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
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()]
|