__init__.py 5.9 KB

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