avatar.py 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. import json
  2. from django.core.exceptions import PermissionDenied, ValidationError
  3. from django.utils.translation import ugettext as _
  4. from rest_framework import status
  5. from rest_framework.response import Response
  6. from misago.conf import settings
  7. from misago.core.utils import format_plaintext_for_html
  8. from ... import avatars
  9. from ...forms.moderation import ModerateAvatarForm
  10. def avatar_endpoint(request, pk=None):
  11. if request.user.is_avatar_locked:
  12. if request.user.avatar_lock_user_message:
  13. reason = format_plaintext_for_html(
  14. request.user.avatar_lock_user_message)
  15. else:
  16. reason = None
  17. return Response({
  18. 'detail': _("Your avatar is locked. You can't change it."),
  19. 'reason': reason
  20. },
  21. status=status.HTTP_403_FORBIDDEN)
  22. avatar_options = get_avatar_options(request.user)
  23. if request.method == 'POST':
  24. return avatar_post(avatar_options, request.user, request.data)
  25. else:
  26. return Response(avatar_options)
  27. def get_avatar_options(user):
  28. options = {
  29. 'avatar_hash': user.avatar_hash,
  30. 'generated': True,
  31. 'gravatar': False,
  32. 'crop_org': False,
  33. 'crop_tmp': False,
  34. 'upload': False,
  35. 'galleries': False
  36. }
  37. # Allow existing galleries
  38. if avatars.gallery.galleries_exist():
  39. options['galleries'] = avatars.gallery.get_available_galleries()
  40. # Can't have custom avatar?
  41. if not settings.allow_custom_avatars:
  42. return options
  43. # Allow Gravatar download
  44. options['gravatar'] = True
  45. # Get avatar tokens
  46. tokens = avatars.get_user_avatar_tokens(user)
  47. # Allow crop with token if we have uploaded avatar
  48. if avatars.uploaded.has_original_avatar(user):
  49. try:
  50. options['crop_org'] = {
  51. 'secret': tokens['org'],
  52. 'crop': json.loads(user.avatar_crop),
  53. 'size': max(settings.MISAGO_AVATARS_SIZES)
  54. }
  55. except (TypeError, ValueError):
  56. pass
  57. # Allow crop of uploaded avatar
  58. if avatars.uploaded.has_temporary_avatar(user):
  59. options['crop_tmp'] = {
  60. 'secret': tokens['tmp'],
  61. 'size': max(settings.MISAGO_AVATARS_SIZES)
  62. }
  63. # Allow upload conditions
  64. options['upload'] = {
  65. 'limit': settings.avatar_upload_limit * 1000,
  66. 'allowed_extensions': avatars.uploaded.ALLOWED_EXTENSIONS,
  67. 'allowed_mime_types': avatars.uploaded.ALLOWED_MIME_TYPES,
  68. }
  69. return options
  70. class AvatarError(Exception):
  71. pass
  72. def avatar_post(options, user, data):
  73. try:
  74. type_options = options[data.get('avatar', 'nope')]
  75. if not type_options:
  76. return Response({'detail': _("This avatar type is not allowed.")},
  77. status=status.HTTP_400_BAD_REQUEST)
  78. rpc_handler = AVATAR_TYPES[data.get('avatar', 'nope')]
  79. except KeyError:
  80. return Response({'detail': _("Unknown avatar type.")},
  81. status=status.HTTP_400_BAD_REQUEST)
  82. try:
  83. response_dict = {'detail': rpc_handler(user, data)}
  84. except AvatarError as e:
  85. return Response({'detail': e.args[0]},
  86. status=status.HTTP_400_BAD_REQUEST)
  87. user.avatar_hash = avatars.get_avatar_hash(user)
  88. user.save(update_fields=['avatar_hash', 'avatar_crop'])
  89. response_dict['avatar_hash'] = user.avatar_hash
  90. response_dict['options'] = get_avatar_options(user)
  91. return Response(response_dict)
  92. """
  93. Avatar rpc handlers
  94. """
  95. def avatar_generate(user, data):
  96. avatars.dynamic.set_avatar(user)
  97. return _("New avatar based on your account was set.")
  98. def avatar_gravatar(user, data):
  99. try:
  100. avatars.gravatar.set_avatar(user)
  101. return _("Gravatar was downloaded and set as new avatar.")
  102. except avatars.gravatar.GravatarError:
  103. raise AvatarError(_("Failed to connect to Gravatar servers."))
  104. except avatars.gravatar.NoGravatarAvailable:
  105. raise AvatarError(
  106. _("No Gravatar is associated with your e-mail address."))
  107. def avatar_gallery(user, data):
  108. image = data.get('image') or 'not-possible'
  109. if avatars.gallery.is_avatar_from_gallery(image):
  110. avatars.gallery.set_avatar(user, image)
  111. return _("Avatar from gallery was set.")
  112. else:
  113. raise AvatarError(_("Incorrect image."))
  114. def avatar_upload(user, data):
  115. new_avatar = data.get('image')
  116. if not new_avatar:
  117. raise AvatarError(_("No file was sent."))
  118. try:
  119. avatars.uploaded.handle_uploaded_file(user, new_avatar)
  120. except ValidationError as e:
  121. raise AvatarError(e.args[0])
  122. # send back token for temp image
  123. return avatars.get_avatar_hash(user, 'tmp')
  124. def avatar_crop_org(user, data):
  125. avatar_crop(user, data, 'org')
  126. return _("Avatar was re-cropped.")
  127. def avatar_crop_tmp(user, data):
  128. avatar_crop(user, data, 'tmp')
  129. return _("Uploaded avatar was set.")
  130. def avatar_crop(user, data, suffix):
  131. try:
  132. crop = avatars.uploaded.crop_source_image(
  133. user, suffix, data.get('crop', {}))
  134. user.avatar_crop = json.dumps(crop)
  135. except ValidationError as e:
  136. raise AvatarError(e.args[0])
  137. AVATAR_TYPES = {
  138. 'generated': avatar_generate,
  139. 'gravatar': avatar_gravatar,
  140. 'galleries': avatar_gallery,
  141. 'upload': avatar_upload,
  142. 'crop_org': avatar_crop_org,
  143. 'crop_tmp': avatar_crop_tmp,
  144. }
  145. def moderate_avatar_endpoint(request, profile):
  146. if request.method == "POST":
  147. is_avatar_locked = profile.is_avatar_locked
  148. form = ModerateAvatarForm(request.data, instance=profile)
  149. if form.is_valid():
  150. if form.cleaned_data['is_avatar_locked'] and not is_avatar_locked:
  151. avatars.dynamic.set_avatar(profile)
  152. profile.avatar_hash = avatars.get_avatar_hash(profile)
  153. form.save()
  154. return Response({
  155. 'avatar_hash': profile.avatar_hash,
  156. 'is_avatar_locked': int(profile.is_avatar_locked),
  157. 'avatar_lock_user_message': profile.avatar_lock_user_message,
  158. 'avatar_lock_staff_message': profile.avatar_lock_staff_message,
  159. })
  160. else:
  161. return Response(form.errors, status=status.HTTP_400_BAD_REQUEST)
  162. else:
  163. return Response({
  164. 'is_avatar_locked': int(profile.is_avatar_locked),
  165. 'avatar_lock_user_message': profile.avatar_lock_user_message,
  166. 'avatar_lock_staff_message': profile.avatar_lock_staff_message,
  167. })