Development of an internal social media platform with personalised dashboards for students
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

hstore.py 3.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  1. import json
  2. from django.contrib.postgres import forms, lookups
  3. from django.contrib.postgres.fields.array import ArrayField
  4. from django.core import exceptions
  5. from django.db.models import Field, TextField, Transform
  6. from django.utils.translation import gettext_lazy as _
  7. __all__ = ['HStoreField']
  8. class HStoreField(Field):
  9. empty_strings_allowed = False
  10. description = _('Map of strings to strings/nulls')
  11. default_error_messages = {
  12. 'not_a_string': _('The value of "%(key)s" is not a string or null.'),
  13. }
  14. def db_type(self, connection):
  15. return 'hstore'
  16. def get_transform(self, name):
  17. transform = super().get_transform(name)
  18. if transform:
  19. return transform
  20. return KeyTransformFactory(name)
  21. def validate(self, value, model_instance):
  22. super().validate(value, model_instance)
  23. for key, val in value.items():
  24. if not isinstance(val, str) and val is not None:
  25. raise exceptions.ValidationError(
  26. self.error_messages['not_a_string'],
  27. code='not_a_string',
  28. params={'key': key},
  29. )
  30. def to_python(self, value):
  31. if isinstance(value, str):
  32. value = json.loads(value)
  33. return value
  34. def value_to_string(self, obj):
  35. return json.dumps(self.value_from_object(obj))
  36. def formfield(self, **kwargs):
  37. defaults = {
  38. 'form_class': forms.HStoreField,
  39. }
  40. defaults.update(kwargs)
  41. return super().formfield(**defaults)
  42. def get_prep_value(self, value):
  43. value = super().get_prep_value(value)
  44. if isinstance(value, dict):
  45. prep_value = {}
  46. for key, val in value.items():
  47. key = str(key)
  48. if val is not None:
  49. val = str(val)
  50. prep_value[key] = val
  51. value = prep_value
  52. if isinstance(value, list):
  53. value = [str(item) for item in value]
  54. return value
  55. HStoreField.register_lookup(lookups.DataContains)
  56. HStoreField.register_lookup(lookups.ContainedBy)
  57. HStoreField.register_lookup(lookups.HasKey)
  58. HStoreField.register_lookup(lookups.HasKeys)
  59. HStoreField.register_lookup(lookups.HasAnyKeys)
  60. class KeyTransform(Transform):
  61. output_field = TextField()
  62. def __init__(self, key_name, *args, **kwargs):
  63. super().__init__(*args, **kwargs)
  64. self.key_name = key_name
  65. def as_sql(self, compiler, connection):
  66. lhs, params = compiler.compile(self.lhs)
  67. return "(%s -> '%s')" % (lhs, self.key_name), params
  68. class KeyTransformFactory:
  69. def __init__(self, key_name):
  70. self.key_name = key_name
  71. def __call__(self, *args, **kwargs):
  72. return KeyTransform(self.key_name, *args, **kwargs)
  73. @HStoreField.register_lookup
  74. class KeysTransform(Transform):
  75. lookup_name = 'keys'
  76. function = 'akeys'
  77. output_field = ArrayField(TextField())
  78. @HStoreField.register_lookup
  79. class ValuesTransform(Transform):
  80. lookup_name = 'values'
  81. function = 'avals'
  82. output_field = ArrayField(TextField())