__init__.py 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. import copy
  2. import logging
  3. from django.core.exceptions import ValidationError
  4. from django.urls import reverse
  5. from django.utils.module_loading import import_string
  6. from django.utils.translation import gettext as _
  7. from ...conf import settings
  8. from .basefields import *
  9. from .serializers import serialize_profilefields_data
  10. logger = logging.getLogger("misago.users.ProfileFields")
  11. class ProfileFields:
  12. def __init__(self, fields_groups):
  13. self.is_loaded = False
  14. self.fields_groups = fields_groups
  15. self.fields_dict = {}
  16. def load(self):
  17. self.fields_dict = {}
  18. fieldnames = {}
  19. for group in self.fields_groups:
  20. for field_path in group["fields"]:
  21. field = import_string(field_path)
  22. field._field_path = field_path
  23. if field_path in self.fields_dict:
  24. raise ValueError(
  25. "%s profile field has been specified twice" % field._field_path
  26. )
  27. if not getattr(field, "fieldname", None):
  28. raise ValueError(
  29. "%s profile field has to specify fieldname attribute"
  30. % field._field_path
  31. )
  32. if field.fieldname in fieldnames:
  33. raise ValueError(
  34. (
  35. '%s profile field defines fieldname "%s" '
  36. "that is already in use by the %s"
  37. )
  38. % (
  39. field._field_path,
  40. field.fieldname,
  41. fieldnames[field.fieldname],
  42. )
  43. )
  44. fieldnames[field.fieldname] = field_path
  45. self.fields_dict[field_path] = field()
  46. self.is_loaded = True
  47. def get_fields(self):
  48. if not self.is_loaded:
  49. self.load()
  50. return self.fields_dict.values()
  51. def get_fields_groups(self):
  52. if not self.is_loaded:
  53. self.load()
  54. groups = []
  55. for group in self.fields_groups:
  56. group_dict = {"name": _(group["name"]), "fields": []}
  57. for field_path in group["fields"]:
  58. field = self.fields_dict[field_path]
  59. group_dict["fields"].append(field)
  60. if group_dict["fields"]:
  61. groups.append(group_dict)
  62. return groups
  63. def add_fields_to_form(self, request, user, form):
  64. if not self.is_loaded:
  65. self.load()
  66. form._profile_fields = []
  67. for field in self.get_fields():
  68. if not field.is_editable(request, user):
  69. continue
  70. form._profile_fields.append(field.fieldname)
  71. form.fields[field.fieldname] = field.get_form_field(request, user)
  72. def add_fields_to_admin_form(self, request, user, form):
  73. self.add_fields_to_form(request, user, form)
  74. form._profile_fields_groups = []
  75. for group in self.fields_groups:
  76. group_dict = {"name": _(group["name"]), "fields": []}
  77. for field_path in group["fields"]:
  78. field = self.fields_dict[field_path]
  79. if field.fieldname in form._profile_fields:
  80. group_dict["fields"].append(field.fieldname)
  81. if group_dict["fields"]:
  82. form._profile_fields_groups.append(group_dict)
  83. def clean_form(self, request, user, form, cleaned_data):
  84. for field in self.get_fields():
  85. if field.fieldname not in cleaned_data:
  86. continue
  87. try:
  88. cleaned_data[field.fieldname] = field.clean(
  89. request, user, cleaned_data[field.fieldname]
  90. )
  91. except ValidationError as e:
  92. form.add_error(field.fieldname, e)
  93. return cleaned_data
  94. def update_user_profile_fields(self, request, user, form):
  95. old_fields = copy.copy(user.profile_fields or {})
  96. new_fields = {}
  97. for fieldname in form._profile_fields:
  98. if fieldname in form.cleaned_data:
  99. new_fields[fieldname] = form.cleaned_data[fieldname]
  100. user.profile_fields = new_fields
  101. old_fields_reduced = {k: v for k, v in old_fields.items() if v}
  102. new_fields_reduced = {k: v for k, v in new_fields.items() if v}
  103. if old_fields_reduced != new_fields_reduced:
  104. self.log_profile_fields_update(request, user)
  105. def log_profile_fields_update(self, request, user):
  106. if request.user == user:
  107. log_message = "%s edited own profile fields" % user.username
  108. else:
  109. log_message = "%s edited %s's (#%s) profile fields" % (
  110. request.user,
  111. user.username,
  112. user.pk,
  113. )
  114. logger.info(
  115. log_message,
  116. extra={
  117. "absolute_url": request.build_absolute_uri(
  118. reverse(
  119. "misago:user-details", kwargs={"slug": user.slug, "pk": user.pk}
  120. )
  121. )
  122. },
  123. )
  124. def search_users(self, criteria, queryset):
  125. if not self.is_loaded:
  126. self.load()
  127. q_obj = None
  128. for field in self.fields_dict.values():
  129. q = field.search_users(criteria)
  130. if q:
  131. if q_obj:
  132. q_obj = q_obj | q
  133. else:
  134. q_obj = q
  135. if q_obj:
  136. return queryset.filter(q_obj)
  137. return queryset
  138. profilefields = ProfileFields(settings.MISAGO_PROFILE_FIELDS)