2019-03-11 02:06:45 +00:00
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
#
|
2022-11-11 07:04:31 +00:00
|
|
|
|
|
2022-11-11 09:28:13 +00:00
|
|
|
|
from django.core.validators import MinValueValidator, MaxValueValidator
|
2019-03-11 02:06:45 +00:00
|
|
|
|
from django.db import models
|
2020-06-11 04:10:00 +00:00
|
|
|
|
from django.utils.encoding import force_text
|
2022-11-11 09:28:13 +00:00
|
|
|
|
from django.utils.translation import ugettext_lazy as _
|
2023-02-19 07:10:10 +00:00
|
|
|
|
from rest_framework.utils.encoders import JSONEncoder
|
2022-11-11 07:04:31 +00:00
|
|
|
|
|
2022-11-04 06:22:38 +00:00
|
|
|
|
from common.local import add_encrypted_field_set
|
2022-12-05 10:54:12 +00:00
|
|
|
|
from common.utils import signer, crypto
|
2023-01-16 11:02:09 +00:00
|
|
|
|
from .validators import PortRangeValidator
|
2019-03-11 02:06:45 +00:00
|
|
|
|
|
|
|
|
|
__all__ = [
|
2022-11-11 07:11:10 +00:00
|
|
|
|
"JsonMixin",
|
|
|
|
|
"JsonDictMixin",
|
|
|
|
|
"JsonListMixin",
|
|
|
|
|
"JsonTypeMixin",
|
|
|
|
|
"JsonCharField",
|
|
|
|
|
"JsonTextField",
|
|
|
|
|
"JsonListCharField",
|
|
|
|
|
"JsonListTextField",
|
|
|
|
|
"JsonDictCharField",
|
|
|
|
|
"JsonDictTextField",
|
|
|
|
|
"EncryptCharField",
|
|
|
|
|
"EncryptTextField",
|
|
|
|
|
"EncryptMixin",
|
|
|
|
|
"EncryptJsonDictTextField",
|
|
|
|
|
"EncryptJsonDictCharField",
|
|
|
|
|
"PortField",
|
2023-01-16 11:02:09 +00:00
|
|
|
|
"PortRangeField",
|
2022-11-11 07:11:10 +00:00
|
|
|
|
"BitChoices",
|
2023-01-16 11:02:09 +00:00
|
|
|
|
"TreeChoices",
|
2023-04-23 08:15:27 +00:00
|
|
|
|
"JSONManyToManyField",
|
2019-03-11 02:06:45 +00:00
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class JsonMixin:
|
|
|
|
|
tp = None
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
|
def json_decode(data):
|
|
|
|
|
try:
|
|
|
|
|
return json.loads(data)
|
|
|
|
|
except (TypeError, json.JSONDecodeError):
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
|
def json_encode(data):
|
2023-02-19 07:10:10 +00:00
|
|
|
|
return json.dumps(data, cls=JSONEncoder)
|
2019-03-11 02:06:45 +00:00
|
|
|
|
|
2020-11-19 07:50:31 +00:00
|
|
|
|
def from_db_value(self, value, expression, connection, context=None):
|
2019-03-11 02:06:45 +00:00
|
|
|
|
if value is None:
|
|
|
|
|
return value
|
|
|
|
|
return self.json_decode(value)
|
|
|
|
|
|
|
|
|
|
def to_python(self, value):
|
|
|
|
|
if value is None:
|
|
|
|
|
return value
|
|
|
|
|
|
|
|
|
|
if not isinstance(value, str) or not value.startswith('"'):
|
|
|
|
|
return value
|
|
|
|
|
else:
|
|
|
|
|
return self.json_decode(value)
|
|
|
|
|
|
|
|
|
|
def get_prep_value(self, value):
|
|
|
|
|
if value is None:
|
|
|
|
|
return value
|
|
|
|
|
return self.json_encode(value)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class JsonTypeMixin(JsonMixin):
|
|
|
|
|
tp = dict
|
|
|
|
|
|
2020-11-19 07:50:31 +00:00
|
|
|
|
def from_db_value(self, value, expression, connection, context=None):
|
2019-03-11 02:06:45 +00:00
|
|
|
|
value = super().from_db_value(value, expression, connection, context)
|
|
|
|
|
if not isinstance(value, self.tp):
|
|
|
|
|
value = self.tp()
|
|
|
|
|
return value
|
|
|
|
|
|
|
|
|
|
def to_python(self, value):
|
|
|
|
|
data = super().to_python(value)
|
|
|
|
|
if not isinstance(data, self.tp):
|
|
|
|
|
data = self.tp()
|
|
|
|
|
return data
|
|
|
|
|
|
|
|
|
|
def get_prep_value(self, value):
|
|
|
|
|
if not isinstance(value, self.tp):
|
|
|
|
|
value = self.tp()
|
|
|
|
|
return self.json_encode(value)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class JsonDictMixin(JsonTypeMixin):
|
|
|
|
|
tp = dict
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class JsonDictCharField(JsonDictMixin, models.CharField):
|
|
|
|
|
description = _("Marshal dict data to char field")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class JsonDictTextField(JsonDictMixin, models.TextField):
|
|
|
|
|
description = _("Marshal dict data to text field")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class JsonListMixin(JsonTypeMixin):
|
|
|
|
|
tp = list
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class JsonStrListMixin(JsonListMixin):
|
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class JsonListCharField(JsonListMixin, models.CharField):
|
|
|
|
|
description = _("Marshal list data to char field")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class JsonListTextField(JsonListMixin, models.TextField):
|
|
|
|
|
description = _("Marshal list data to text field")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class JsonCharField(JsonMixin, models.CharField):
|
|
|
|
|
description = _("Marshal data to char field")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class JsonTextField(JsonMixin, models.TextField):
|
|
|
|
|
description = _("Marshal data to text field")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class EncryptMixin:
|
2019-12-18 07:37:53 +00:00
|
|
|
|
"""
|
|
|
|
|
EncryptMixin要放在最前面
|
|
|
|
|
"""
|
2020-06-11 06:10:55 +00:00
|
|
|
|
|
|
|
|
|
def decrypt_from_signer(self, value):
|
2022-11-11 07:11:10 +00:00
|
|
|
|
return signer.unsign(value) or ""
|
2020-06-11 06:10:55 +00:00
|
|
|
|
|
2020-11-19 07:50:31 +00:00
|
|
|
|
def from_db_value(self, value, expression, connection, context=None):
|
2022-12-05 10:54:12 +00:00
|
|
|
|
if value is None:
|
2019-12-18 07:37:53 +00:00
|
|
|
|
return value
|
2020-06-11 04:10:00 +00:00
|
|
|
|
|
2022-10-18 12:37:17 +00:00
|
|
|
|
value = force_text(value)
|
2020-10-28 09:28:15 +00:00
|
|
|
|
plain_value = crypto.decrypt(value)
|
2020-06-11 04:10:00 +00:00
|
|
|
|
|
|
|
|
|
# 如果没有解开,使用原来的signer解密
|
|
|
|
|
if not plain_value:
|
2020-06-11 06:10:55 +00:00
|
|
|
|
plain_value = self.decrypt_from_signer(value)
|
|
|
|
|
|
|
|
|
|
# 可能和Json mix,所以要先解密,再json
|
2019-12-18 07:37:53 +00:00
|
|
|
|
sp = super()
|
2022-11-11 07:11:10 +00:00
|
|
|
|
if hasattr(sp, "from_db_value"):
|
2020-06-11 04:10:00 +00:00
|
|
|
|
plain_value = sp.from_db_value(plain_value, expression, connection, context)
|
|
|
|
|
return plain_value
|
2019-03-11 02:06:45 +00:00
|
|
|
|
|
|
|
|
|
def get_prep_value(self, value):
|
2022-12-05 10:54:12 +00:00
|
|
|
|
if value is None:
|
2019-03-11 02:06:45 +00:00
|
|
|
|
return value
|
2020-06-11 06:10:55 +00:00
|
|
|
|
|
|
|
|
|
# 先 json 再解密
|
2019-12-18 07:37:53 +00:00
|
|
|
|
sp = super()
|
2022-11-11 07:11:10 +00:00
|
|
|
|
if hasattr(sp, "get_prep_value"):
|
2019-12-18 07:37:53 +00:00
|
|
|
|
value = sp.get_prep_value(value)
|
2020-06-11 04:10:00 +00:00
|
|
|
|
value = force_text(value)
|
|
|
|
|
# 替换新的加密方式
|
2020-10-28 09:28:15 +00:00
|
|
|
|
return crypto.encrypt(value)
|
2019-03-11 02:06:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class EncryptTextField(EncryptMixin, models.TextField):
|
|
|
|
|
description = _("Encrypt field using Secret Key")
|
|
|
|
|
|
2022-11-04 06:22:38 +00:00
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
|
add_encrypted_field_set(self.verbose_name)
|
|
|
|
|
|
2019-03-11 02:06:45 +00:00
|
|
|
|
|
|
|
|
|
class EncryptCharField(EncryptMixin, models.CharField):
|
2019-06-24 14:16:39 +00:00
|
|
|
|
@staticmethod
|
|
|
|
|
def change_max_length(kwargs):
|
2022-11-11 07:11:10 +00:00
|
|
|
|
kwargs.setdefault("max_length", 1024)
|
|
|
|
|
max_length = kwargs.get("max_length")
|
2019-06-24 14:16:39 +00:00
|
|
|
|
if max_length < 129:
|
|
|
|
|
max_length = 128
|
|
|
|
|
max_length = max_length * 2
|
2022-11-11 07:11:10 +00:00
|
|
|
|
kwargs["max_length"] = max_length
|
2019-06-24 14:16:39 +00:00
|
|
|
|
|
2019-03-11 02:06:45 +00:00
|
|
|
|
def __init__(self, *args, **kwargs):
|
2019-06-24 14:16:39 +00:00
|
|
|
|
self.change_max_length(kwargs)
|
2019-03-11 02:06:45 +00:00
|
|
|
|
super().__init__(*args, **kwargs)
|
2022-11-04 06:22:38 +00:00
|
|
|
|
add_encrypted_field_set(self.verbose_name)
|
2019-03-11 02:06:45 +00:00
|
|
|
|
|
2019-06-24 14:16:39 +00:00
|
|
|
|
def deconstruct(self):
|
|
|
|
|
name, path, args, kwargs = super().deconstruct()
|
2022-11-11 07:11:10 +00:00
|
|
|
|
max_length = kwargs.pop("max_length")
|
2019-06-24 14:16:39 +00:00
|
|
|
|
if max_length > 255:
|
|
|
|
|
max_length = max_length // 2
|
2022-11-11 07:11:10 +00:00
|
|
|
|
kwargs["max_length"] = max_length
|
2019-06-24 14:16:39 +00:00
|
|
|
|
return name, path, args, kwargs
|
|
|
|
|
|
2019-03-11 02:06:45 +00:00
|
|
|
|
|
2019-05-20 11:39:53 +00:00
|
|
|
|
class EncryptJsonDictTextField(EncryptMixin, JsonDictTextField):
|
2022-11-04 06:22:38 +00:00
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
|
add_encrypted_field_set(self.verbose_name)
|
2019-05-20 11:39:53 +00:00
|
|
|
|
|
2019-03-11 02:06:45 +00:00
|
|
|
|
|
2019-12-18 07:37:53 +00:00
|
|
|
|
class EncryptJsonDictCharField(EncryptMixin, JsonDictCharField):
|
2022-11-04 06:22:38 +00:00
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
|
add_encrypted_field_set(self.verbose_name)
|
2019-12-18 07:37:53 +00:00
|
|
|
|
|
2022-04-12 09:45:10 +00:00
|
|
|
|
|
|
|
|
|
class PortField(models.IntegerField):
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
2022-11-11 07:11:10 +00:00
|
|
|
|
kwargs.update(
|
|
|
|
|
{
|
|
|
|
|
"blank": False,
|
|
|
|
|
"null": False,
|
|
|
|
|
"validators": [MinValueValidator(0), MaxValueValidator(65535)],
|
|
|
|
|
}
|
|
|
|
|
)
|
2022-04-12 09:45:10 +00:00
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
|
|
2022-11-11 07:04:31 +00:00
|
|
|
|
|
2023-01-16 11:02:09 +00:00
|
|
|
|
class TreeChoices(models.Choices):
|
2022-11-11 07:04:31 +00:00
|
|
|
|
@classmethod
|
2023-01-16 11:02:09 +00:00
|
|
|
|
def is_tree(cls):
|
|
|
|
|
return True
|
2022-11-11 07:04:31 +00:00
|
|
|
|
|
2022-11-11 09:28:13 +00:00
|
|
|
|
@classmethod
|
2023-01-16 11:02:09 +00:00
|
|
|
|
def branches(cls):
|
|
|
|
|
return [i for i in cls]
|
2022-11-11 09:28:13 +00:00
|
|
|
|
|
2022-11-11 07:04:31 +00:00
|
|
|
|
@classmethod
|
|
|
|
|
def tree(cls):
|
2022-11-11 09:28:13 +00:00
|
|
|
|
if not cls.is_tree():
|
|
|
|
|
return []
|
2022-11-11 07:11:10 +00:00
|
|
|
|
root = [_("All"), cls.branches()]
|
2022-11-11 09:28:13 +00:00
|
|
|
|
return [cls.render_node(root)]
|
2022-11-11 07:04:31 +00:00
|
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
|
def render_node(cls, node):
|
2023-01-16 11:02:09 +00:00
|
|
|
|
if isinstance(node, models.Choices):
|
2022-11-11 07:04:31 +00:00
|
|
|
|
return {
|
2022-11-11 09:28:13 +00:00
|
|
|
|
"value": node.name,
|
2022-11-11 07:11:10 +00:00
|
|
|
|
"label": node.label,
|
2022-11-11 07:04:31 +00:00
|
|
|
|
}
|
|
|
|
|
else:
|
|
|
|
|
name, children = node
|
|
|
|
|
return {
|
2022-11-11 09:28:13 +00:00
|
|
|
|
"value": name,
|
2022-11-11 07:11:10 +00:00
|
|
|
|
"label": name,
|
|
|
|
|
"children": [cls.render_node(child) for child in children],
|
2022-11-11 07:04:31 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-01-16 11:02:09 +00:00
|
|
|
|
@classmethod
|
|
|
|
|
def all(cls):
|
|
|
|
|
return [i[0] for i in cls.choices]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class BitChoices(models.IntegerChoices, TreeChoices):
|
|
|
|
|
@classmethod
|
|
|
|
|
def is_tree(cls):
|
|
|
|
|
return False
|
|
|
|
|
|
2022-11-11 07:04:31 +00:00
|
|
|
|
@classmethod
|
|
|
|
|
def all(cls):
|
|
|
|
|
value = 0
|
|
|
|
|
for c in cls:
|
|
|
|
|
value |= c.value
|
|
|
|
|
return value
|
2023-01-16 11:02:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class PortRangeField(models.CharField):
|
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
|
kwargs['max_length'] = 16
|
|
|
|
|
super().__init__(**kwargs)
|
|
|
|
|
self.validators.append(PortRangeValidator())
|
2023-04-23 08:15:27 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
from django.db.models import Q
|
|
|
|
|
from django.apps import apps
|
|
|
|
|
|
|
|
|
|
from django.db import models
|
|
|
|
|
from django.core.exceptions import ValidationError
|
|
|
|
|
import json
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class JSONManyToManyDescriptor:
|
|
|
|
|
def __init__(self, field):
|
|
|
|
|
self.field = field
|
|
|
|
|
self._is_setting = False
|
|
|
|
|
|
|
|
|
|
def __get__(self, instance, owner=None):
|
|
|
|
|
if instance is None:
|
|
|
|
|
return self
|
|
|
|
|
|
|
|
|
|
if not hasattr(instance, "_related_manager_cache"):
|
|
|
|
|
instance._related_manager_cache = {}
|
|
|
|
|
|
|
|
|
|
current_value = getattr(instance, self.field.attname, {})
|
|
|
|
|
|
|
|
|
|
if self.field.name not in instance._related_manager_cache or instance._related_manager_cache[
|
|
|
|
|
self.field.name]._is_value_stale(current_value):
|
|
|
|
|
manager = RelatedManager(instance, self.field)
|
|
|
|
|
instance._related_manager_cache[self.field.name] = manager
|
|
|
|
|
|
|
|
|
|
return instance._related_manager_cache[self.field.name]
|
|
|
|
|
|
|
|
|
|
def __set__(self, instance, value):
|
|
|
|
|
if instance is None:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
if not hasattr(instance, "_is_setting"):
|
|
|
|
|
instance._is_setting = {}
|
|
|
|
|
|
|
|
|
|
if self.field.name not in instance._is_setting or not instance._is_setting[self.field.name]:
|
|
|
|
|
instance._is_setting[self.field.name] = True
|
|
|
|
|
manager = self.__get__(instance, instance.__class__)
|
|
|
|
|
manager.set(value)
|
|
|
|
|
serialized_value = manager.serialize()
|
|
|
|
|
instance.__dict__[self.field.attname] = serialized_value
|
|
|
|
|
instance._is_setting[self.field.name] = False
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class JSONManyToManyField(models.JSONField):
|
|
|
|
|
def __init__(self, related_model, *args, **kwargs):
|
|
|
|
|
self.related_model = related_model
|
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
|
|
|
|
|
|
def contribute_to_class(self, cls, name, **kwargs):
|
|
|
|
|
super().contribute_to_class(cls, name, **kwargs)
|
|
|
|
|
setattr(cls, self.name, JSONManyToManyDescriptor(self))
|
|
|
|
|
|
|
|
|
|
def deconstruct(self):
|
|
|
|
|
name, path, args, kwargs = super().deconstruct()
|
|
|
|
|
kwargs['related_model'] = self.related_model
|
|
|
|
|
return name, path, args, kwargs
|
|
|
|
|
|
|
|
|
|
def validate(self, value, model_instance):
|
|
|
|
|
super().validate(value, model_instance)
|
|
|
|
|
if not isinstance(value, list) or not all(isinstance(item, int) for item in value):
|
|
|
|
|
raise ValidationError("Invalid JSON data for JSONManyToManyField.")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class RelatedManager:
|
|
|
|
|
def __init__(self, instance, field):
|
|
|
|
|
self.instance = instance
|
|
|
|
|
self.field = field
|
|
|
|
|
|
|
|
|
|
def _is_value_stale(self, current_value):
|
|
|
|
|
return self.serialize() != current_value
|
|
|
|
|
|
|
|
|
|
def set(self, value):
|
|
|
|
|
self.field.value = value
|
|
|
|
|
|
|
|
|
|
def serialize(self):
|
|
|
|
|
return self.field.value
|
|
|
|
|
|
|
|
|
|
def _get_queryset(self):
|
|
|
|
|
model = apps.get_model(self.field.to)
|
|
|
|
|
value = self.field.value
|
|
|
|
|
|
|
|
|
|
if value["type"] == "all":
|
|
|
|
|
return model.objects.all()
|
|
|
|
|
elif value["type"] == "ids":
|
|
|
|
|
return model.objects.filter(id__in=value["ids"])
|
|
|
|
|
elif value["type"] == "attrs":
|
|
|
|
|
filters = Q()
|
|
|
|
|
for attr in value["attrs"]:
|
|
|
|
|
if attr["match"] == "exact":
|
|
|
|
|
filters &= Q(**{attr["attr"]: attr["value"]})
|
|
|
|
|
return model.objects.filter(filters)
|
|
|
|
|
|
|
|
|
|
def all(self):
|
|
|
|
|
return self._get_queryset()
|
|
|
|
|
|
|
|
|
|
def filter(self, *args, **kwargs):
|
|
|
|
|
queryset = self._get_queryset()
|
|
|
|
|
return queryset.filter(*args, **kwargs)
|