__init__.py 5.7 KB

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