2018-11-03 23:40:20 +01:00
|
|
|
from typing import Any, Dict, List, Optional
|
2017-08-24 12:26:55 +02:00
|
|
|
|
|
|
|
from django.core.exceptions import ImproperlyConfigured
|
2012-07-07 11:14:04 +02:00
|
|
|
from django.db import models
|
|
|
|
|
2018-07-09 23:22:26 +02:00
|
|
|
from .access_permissions import BaseAccessPermissions
|
2019-01-19 14:02:13 +01:00
|
|
|
from .autoupdate import Element, inform_changed_data, inform_changed_elements
|
2018-11-05 09:04:41 +01:00
|
|
|
from .rest_api import model_serializer_classes
|
2017-01-18 16:29:13 +01:00
|
|
|
from .utils import convert_camel_case_to_pseudo_snake_case
|
|
|
|
|
2012-11-24 14:01:21 +01:00
|
|
|
|
2012-07-07 11:14:04 +02:00
|
|
|
class MinMaxIntegerField(models.IntegerField):
|
2013-09-25 12:53:44 +02:00
|
|
|
"""
|
|
|
|
IntegerField with options to set a min- and a max-value.
|
|
|
|
"""
|
|
|
|
|
2019-01-06 16:22:33 +01:00
|
|
|
def __init__(
|
|
|
|
self, min_value: int = None, max_value: int = None, *args: Any, **kwargs: Any
|
|
|
|
) -> None:
|
2012-07-07 11:14:04 +02:00
|
|
|
self.min_value, self.max_value = min_value, max_value
|
|
|
|
super(MinMaxIntegerField, self).__init__(*args, **kwargs)
|
|
|
|
|
2017-08-24 12:26:55 +02:00
|
|
|
def formfield(self, **kwargs: Any) -> Any:
|
2019-01-06 16:22:33 +01:00
|
|
|
defaults = {"min_value": self.min_value, "max_value": self.max_value}
|
2012-07-07 11:14:04 +02:00
|
|
|
defaults.update(kwargs)
|
|
|
|
return super(MinMaxIntegerField, self).formfield(**defaults)
|
2015-06-29 12:08:15 +02:00
|
|
|
|
|
|
|
|
|
|
|
class RESTModelMixin:
|
|
|
|
"""
|
2016-02-11 22:58:32 +01:00
|
|
|
Mixin for Django models which are used in our REST API.
|
2015-06-29 12:08:15 +02:00
|
|
|
"""
|
|
|
|
|
2018-08-22 22:00:08 +02:00
|
|
|
access_permissions: Optional[BaseAccessPermissions] = None
|
2016-02-11 11:29:19 +01:00
|
|
|
|
2017-08-24 12:26:55 +02:00
|
|
|
def get_root_rest_element(self) -> models.Model:
|
2015-06-29 12:08:15 +02:00
|
|
|
"""
|
|
|
|
Returns the root rest instance.
|
|
|
|
|
|
|
|
Uses self as default.
|
|
|
|
"""
|
|
|
|
return self
|
|
|
|
|
2016-09-17 22:26:23 +02:00
|
|
|
@classmethod
|
2017-08-24 12:26:55 +02:00
|
|
|
def get_access_permissions(cls) -> BaseAccessPermissions:
|
2016-02-11 22:58:32 +01:00
|
|
|
"""
|
|
|
|
Returns a container to handle access permissions for this model and
|
|
|
|
its corresponding viewset.
|
|
|
|
"""
|
2017-08-24 12:26:55 +02:00
|
|
|
if cls.access_permissions is None:
|
2019-01-06 16:22:33 +01:00
|
|
|
raise ImproperlyConfigured(
|
|
|
|
"A RESTModel needs to have an access_permission."
|
|
|
|
)
|
2016-09-17 22:26:23 +02:00
|
|
|
return cls.access_permissions
|
2016-02-11 22:58:32 +01:00
|
|
|
|
|
|
|
@classmethod
|
2017-08-24 12:26:55 +02:00
|
|
|
def get_collection_string(cls) -> str:
|
2015-06-29 12:08:15 +02:00
|
|
|
"""
|
2016-02-11 22:58:32 +01:00
|
|
|
Returns the string representing the name of the collection. Returns
|
|
|
|
None if this is not a so called root rest instance.
|
2015-06-29 12:08:15 +02:00
|
|
|
"""
|
2016-02-11 22:58:32 +01:00
|
|
|
# TODO Check if this is a root rest element class and return None if not.
|
2017-08-24 12:26:55 +02:00
|
|
|
app_label = cls._meta.app_label # type: ignore
|
|
|
|
object_name = cls._meta.object_name # type: ignore
|
2019-01-06 16:22:33 +01:00
|
|
|
return "/".join(
|
|
|
|
(
|
|
|
|
convert_camel_case_to_pseudo_snake_case(app_label),
|
|
|
|
convert_camel_case_to_pseudo_snake_case(object_name),
|
|
|
|
)
|
|
|
|
)
|
2016-01-03 15:33:51 +01:00
|
|
|
|
2017-08-24 12:26:55 +02:00
|
|
|
def get_rest_pk(self) -> int:
|
2016-01-03 15:33:51 +01:00
|
|
|
"""
|
2016-02-11 22:58:32 +01:00
|
|
|
Returns the primary key used in the REST API. By default this is
|
|
|
|
the database pk.
|
2016-01-03 15:33:51 +01:00
|
|
|
"""
|
2017-08-24 12:26:55 +02:00
|
|
|
return self.pk # type: ignore
|
2016-09-18 16:00:31 +02:00
|
|
|
|
2018-08-22 07:59:22 +02:00
|
|
|
def save(self, skip_autoupdate: bool = False, *args: Any, **kwargs: Any) -> Any:
|
2016-09-18 16:00:31 +02:00
|
|
|
"""
|
2016-09-30 20:42:58 +02:00
|
|
|
Calls Django's save() method and afterwards hits the autoupdate system.
|
2016-09-18 16:00:31 +02:00
|
|
|
|
|
|
|
If skip_autoupdate is set to True, then the autoupdate system is not
|
|
|
|
informed about the model changed. This also means, that the model cache
|
2018-11-03 23:40:20 +01:00
|
|
|
is not updated. You have to do this manually by calling
|
|
|
|
inform_changed_data().
|
2016-09-18 16:00:31 +02:00
|
|
|
"""
|
2016-09-30 21:43:22 +02:00
|
|
|
# We don't know how to fix this circular import
|
2016-09-18 16:00:31 +02:00
|
|
|
from .autoupdate import inform_changed_data
|
2019-01-06 16:22:33 +01:00
|
|
|
|
2017-08-24 12:26:55 +02:00
|
|
|
return_value = super().save(*args, **kwargs) # type: ignore
|
2016-09-30 20:42:58 +02:00
|
|
|
if not skip_autoupdate:
|
2018-08-22 07:59:22 +02:00
|
|
|
inform_changed_data(self.get_root_rest_element())
|
2016-09-18 16:00:31 +02:00
|
|
|
return return_value
|
|
|
|
|
2018-08-22 07:59:22 +02:00
|
|
|
def delete(self, skip_autoupdate: bool = False, *args: Any, **kwargs: Any) -> Any:
|
2016-09-18 16:00:31 +02:00
|
|
|
"""
|
2016-09-30 20:42:58 +02:00
|
|
|
Calls Django's delete() method and afterwards hits the autoupdate system.
|
2016-09-18 16:00:31 +02:00
|
|
|
|
2016-09-30 21:43:22 +02:00
|
|
|
If skip_autoupdate is set to True, then the autoupdate system is not
|
|
|
|
informed about the model changed. This also means, that the model cache
|
2018-11-03 23:40:20 +01:00
|
|
|
is not updated. You have to do this manually by calling
|
|
|
|
inform_deleted_data().
|
2016-09-18 16:00:31 +02:00
|
|
|
"""
|
2016-09-30 21:43:22 +02:00
|
|
|
# We don't know how to fix this circular import
|
2016-09-18 16:00:31 +02:00
|
|
|
from .autoupdate import inform_changed_data, inform_deleted_data
|
2019-01-06 16:22:33 +01:00
|
|
|
|
2017-08-24 12:26:55 +02:00
|
|
|
instance_pk = self.pk # type: ignore
|
|
|
|
return_value = super().delete(*args, **kwargs) # type: ignore
|
2016-09-30 20:42:58 +02:00
|
|
|
if not skip_autoupdate:
|
|
|
|
if self != self.get_root_rest_element():
|
|
|
|
# The deletion of a included element is a change of the root element.
|
2018-08-22 07:59:22 +02:00
|
|
|
inform_changed_data(self.get_root_rest_element())
|
2016-09-30 20:42:58 +02:00
|
|
|
else:
|
2018-08-22 07:59:22 +02:00
|
|
|
inform_deleted_data([(self.get_collection_string(), instance_pk)])
|
2016-09-18 16:00:31 +02:00
|
|
|
return return_value
|
2018-07-09 23:22:26 +02:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def get_elements(cls) -> List[Dict[str, Any]]:
|
|
|
|
"""
|
|
|
|
Returns all elements as full_data.
|
|
|
|
"""
|
|
|
|
# Get the query to receive all data from the database.
|
|
|
|
try:
|
|
|
|
query = cls.objects.get_full_queryset() # type: ignore
|
|
|
|
except AttributeError:
|
|
|
|
# If the model des not have to method get_full_queryset(), then use
|
|
|
|
# the default queryset from django.
|
|
|
|
query = cls.objects # type: ignore
|
|
|
|
|
|
|
|
# Build a dict from the instance id to the full_data
|
2018-11-03 23:40:20 +01:00
|
|
|
return [instance.get_full_data() for instance in query.all()]
|
2018-07-09 23:22:26 +02:00
|
|
|
|
|
|
|
@classmethod
|
2018-11-01 17:30:18 +01:00
|
|
|
async def restrict_elements(
|
2019-01-06 16:22:33 +01:00
|
|
|
cls, user_id: int, elements: List[Dict[str, Any]]
|
|
|
|
) -> List[Dict[str, Any]]:
|
2018-07-09 23:22:26 +02:00
|
|
|
"""
|
|
|
|
Converts a list of elements from full_data to restricted_data.
|
|
|
|
"""
|
2018-11-03 23:40:20 +01:00
|
|
|
return await cls.get_access_permissions().get_restricted_data(elements, user_id)
|
|
|
|
|
|
|
|
def get_full_data(self) -> Dict[str, Any]:
|
|
|
|
"""
|
|
|
|
Returns the full_data of the instance.
|
|
|
|
"""
|
2018-12-17 13:34:17 +01:00
|
|
|
try:
|
|
|
|
serializer_class = model_serializer_classes[type(self)]
|
|
|
|
except KeyError:
|
|
|
|
# Because of the order of imports, it can happen, that the serializer
|
|
|
|
# for a model is not imported yet. Try to guess the name of the
|
|
|
|
# module and import it.
|
|
|
|
module_name = type(self).__module__.rsplit(".", 1)[0] + ".serializers"
|
|
|
|
__import__(module_name)
|
|
|
|
serializer_class = model_serializer_classes[type(self)]
|
2018-11-05 09:04:41 +01:00
|
|
|
return serializer_class(self).data
|
2019-01-19 14:02:13 +01:00
|
|
|
|
|
|
|
|
|
|
|
def SET_NULL_AND_AUTOUPDATE(
|
|
|
|
collector: Any, field: Any, sub_objs: Any, using: Any
|
|
|
|
) -> None:
|
|
|
|
"""
|
|
|
|
Like models.SET_NULL but also informs the autoupdate system about the
|
2019-01-19 14:32:11 +01:00
|
|
|
instance that was reference.
|
2019-01-19 14:02:13 +01:00
|
|
|
"""
|
|
|
|
if len(sub_objs) != 1:
|
|
|
|
raise RuntimeError(
|
2019-01-19 14:32:11 +01:00
|
|
|
"SET_NULL_AND_AUTOUPDATE is used in an invalid usecase. Please report the bug!"
|
2019-01-19 14:02:13 +01:00
|
|
|
)
|
|
|
|
setattr(sub_objs[0], field.name, None)
|
|
|
|
inform_changed_data(sub_objs[0])
|
|
|
|
models.SET_NULL(collector, field, sub_objs, using)
|
|
|
|
|
|
|
|
|
|
|
|
def CASCADE_AND_AUTOUODATE(
|
|
|
|
collector: Any, field: Any, sub_objs: Any, using: Any
|
|
|
|
) -> None:
|
2019-01-19 14:32:11 +01:00
|
|
|
"""
|
2019-02-27 20:16:51 +01:00
|
|
|
Like models.CASCADE but also informs the autoupdate system about the
|
2019-01-19 14:32:11 +01:00
|
|
|
root rest element of the also deleted instance.
|
|
|
|
"""
|
2019-01-19 14:02:13 +01:00
|
|
|
if len(sub_objs) != 1:
|
|
|
|
raise RuntimeError(
|
2019-01-19 14:32:11 +01:00
|
|
|
"CASCADE_AND_AUTOUPDATE is used in an invalid usecase. Please report the bug!"
|
2019-01-19 14:02:13 +01:00
|
|
|
)
|
|
|
|
root_rest_element = sub_objs[0].get_root_rest_element()
|
|
|
|
inform_changed_elements(
|
|
|
|
[
|
|
|
|
Element(
|
|
|
|
collection_string=root_rest_element.get_collection_string(),
|
|
|
|
id=root_rest_element.pk,
|
|
|
|
full_data=None,
|
|
|
|
reload=True,
|
|
|
|
)
|
|
|
|
]
|
|
|
|
)
|
|
|
|
models.CASCADE(collector, field, sub_objs, using)
|