avatar.py 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  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. from ...models import AvatarGallery
  11. def avatar_endpoint(request, pk=None):
  12. if request.user.is_avatar_locked:
  13. if request.user.avatar_lock_user_message:
  14. reason = format_plaintext_for_html(
  15. request.user.avatar_lock_user_message)
  16. else:
  17. reason = None
  18. return Response({
  19. 'detail': _("Your avatar is locked. You can't change it."),
  20. 'reason': reason
  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. 'avatars': user.avatars,
  30. 'generated': True,
  31. 'gravatar': False,
  32. 'crop_src': 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'] = []
  40. for gallery in avatars.gallery.get_available_galleries():
  41. gallery_images = []
  42. for image in gallery['images']:
  43. gallery_images.append({
  44. 'id': image.id,
  45. 'url': image.url
  46. })
  47. options['galleries'].append({
  48. 'name': gallery['name'],
  49. 'images': gallery_images
  50. })
  51. # Can't have custom avatar?
  52. if not settings.allow_custom_avatars:
  53. return options
  54. # Allow Gravatar download
  55. options['gravatar'] = True
  56. # Allow crop if we have uploaded temporary avatar
  57. if avatars.uploaded.has_source_avatar(user):
  58. try:
  59. options['crop_src'] = {
  60. 'url': user.avatar_src.url,
  61. 'crop': json.loads(user.avatar_crop),
  62. 'size': max(settings.MISAGO_AVATARS_SIZES)
  63. }
  64. except (TypeError, ValueError):
  65. pass
  66. # Allow crop of uploaded avatar
  67. if avatars.uploaded.has_temporary_avatar(user):
  68. options['crop_tmp'] = {
  69. 'url': user.avatar_tmp.url,
  70. 'size': max(settings.MISAGO_AVATARS_SIZES)
  71. }
  72. # Allow upload conditions
  73. options['upload'] = {
  74. 'limit': settings.avatar_upload_limit * 1024,
  75. 'allowed_extensions': avatars.uploaded.ALLOWED_EXTENSIONS,
  76. 'allowed_mime_types': avatars.uploaded.ALLOWED_MIME_TYPES,
  77. }
  78. return options
  79. class AvatarError(Exception):
  80. pass
  81. def avatar_post(options, user, data):
  82. try:
  83. type_options = options[data.get('avatar', 'nope')]
  84. if not type_options:
  85. return Response({
  86. 'detail': _("This avatar type is not allowed.")
  87. }, status=status.HTTP_400_BAD_REQUEST)
  88. rpc_handler = AVATAR_TYPES[data.get('avatar', 'nope')]
  89. except KeyError:
  90. return Response({
  91. 'detail': _("Unknown avatar type.")
  92. }, status=status.HTTP_400_BAD_REQUEST)
  93. try:
  94. response_dict = {'detail': rpc_handler(user, data)}
  95. except AvatarError as e:
  96. return Response({
  97. 'detail': e.args[0]
  98. }, status=status.HTTP_400_BAD_REQUEST)
  99. user.save()
  100. response_dict.update(get_avatar_options(user))
  101. return Response(response_dict)
  102. """
  103. Avatar rpc handlers
  104. """
  105. def avatar_generate(user, data):
  106. avatars.dynamic.set_avatar(user)
  107. return _("New avatar based on your account was set.")
  108. def avatar_gravatar(user, data):
  109. try:
  110. avatars.gravatar.set_avatar(user)
  111. return _("Gravatar was downloaded and set as new avatar.")
  112. except avatars.gravatar.NoGravatarAvailable:
  113. raise AvatarError(
  114. _("No Gravatar is associated with your e-mail address."))
  115. except avatars.gravatar.GravatarError:
  116. raise AvatarError(_("Failed to connect to Gravatar servers."))
  117. def avatar_gallery(user, data):
  118. try:
  119. image_pk = int(data.get('image'))
  120. image = AvatarGallery.objects.get(pk=image_pk)
  121. if image.gallery == '__default__':
  122. raise ValueError()
  123. avatars.gallery.set_avatar(user, image)
  124. return _("Avatar from gallery was set.")
  125. except (TypeError, ValueError, AvatarGallery.DoesNotExist):
  126. raise AvatarError(_("Incorrect image."))
  127. def avatar_upload(user, data):
  128. new_avatar = data.get('image')
  129. if not new_avatar:
  130. raise AvatarError(_("No file was sent."))
  131. try:
  132. avatars.uploaded.handle_uploaded_file(user, new_avatar)
  133. except ValidationError as e:
  134. raise AvatarError(e.args[0])
  135. # send back url for temp image
  136. return user.avatar_tmp.url
  137. def avatar_crop_src(user, data):
  138. avatar_crop(user, data, 'src')
  139. return _("Avatar was re-cropped.")
  140. def avatar_crop_tmp(user, data):
  141. avatar_crop(user, data, 'tmp')
  142. return _("Uploaded avatar was set.")
  143. def avatar_crop(user, data, suffix):
  144. try:
  145. crop = avatars.uploaded.crop_source_image(
  146. user, suffix, data.get('crop', {}))
  147. user.avatar_crop = json.dumps(crop)
  148. except ValidationError as e:
  149. raise AvatarError(e.args[0])
  150. AVATAR_TYPES = {
  151. 'generated': avatar_generate,
  152. 'gravatar': avatar_gravatar,
  153. 'galleries': avatar_gallery,
  154. 'upload': avatar_upload,
  155. 'crop_src': avatar_crop_src,
  156. 'crop_tmp': avatar_crop_tmp,
  157. }
  158. def moderate_avatar_endpoint(request, profile):
  159. if request.method == "POST":
  160. is_avatar_locked = profile.is_avatar_locked
  161. form = ModerateAvatarForm(request.data, instance=profile)
  162. if form.is_valid():
  163. if form.cleaned_data['is_avatar_locked'] and not is_avatar_locked:
  164. avatars.dynamic.set_avatar(profile)
  165. form.save()
  166. return Response({
  167. 'avatars': profile.avatars,
  168. 'is_avatar_locked': int(profile.is_avatar_locked),
  169. 'avatar_lock_user_message': profile.avatar_lock_user_message,
  170. 'avatar_lock_staff_message': profile.avatar_lock_staff_message,
  171. })
  172. else:
  173. return Response(form.errors, status=status.HTTP_400_BAD_REQUEST)
  174. else:
  175. return Response({
  176. 'is_avatar_locked': int(profile.is_avatar_locked),
  177. 'avatar_lock_user_message': profile.avatar_lock_user_message,
  178. 'avatar_lock_staff_message': profile.avatar_lock_staff_message,
  179. })