uploaded.py 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. from hashlib import sha256
  2. from path import path
  3. from PIL import Image
  4. from django.core.exceptions import ValidationError
  5. from django.utils.translation import ugettext as _
  6. from misago.conf import settings
  7. from misago.users.avatars import store
  8. ALLOWED_EXTENSIONS = ('.gif', '.png', '.jpg', '.jpeg')
  9. ALLOWED_MIME_TYPES = ('image/gif', 'image/jpeg', 'image/png')
  10. def validate_file_size(uploaded_file):
  11. upload_limit = settings.avatar_upload_limit * 1024
  12. if uploaded_file.size > upload_limit:
  13. raise ValidationError(_("Uploaded file is too big."))
  14. def validate_extension(uploaded_file):
  15. lowercased_name = uploaded_file.name.lower()
  16. for extension in ALLOWED_EXTENSIONS:
  17. if lowercased_name.endswith(extension):
  18. return True
  19. else:
  20. raise ValidationError(_("Uploaded file type is not allowed."))
  21. def validate_mime(uploaded_file):
  22. if uploaded_file.content_type not in ALLOWED_MIME_TYPES:
  23. raise ValidationError(_("Uploaded file type is not allowed."))
  24. def validate_dimensions(uploaded_file):
  25. image = Image.open(uploaded_file)
  26. if min(image.size) < 100:
  27. message = _("Uploaded image should be at "
  28. "least 100 pixels tall and wide.")
  29. raise ValidationError(message)
  30. image_ratio = float(min(image.size)) / float(max(image.size))
  31. if image_ratio < 0.15:
  32. message = _("Uploaded image ratio cannot be greater than 16:9.")
  33. raise ValidationError(message)
  34. return image
  35. def validate_uploaded_file(uploaded_file):
  36. try:
  37. validate_file_size(uploaded_file)
  38. validate_extension(uploaded_file)
  39. validate_mime(uploaded_file)
  40. return validate_dimensions(uploaded_file)
  41. except ValidationError as e:
  42. try:
  43. temporary_file_path = path(uploaded_file.temporary_file_path())
  44. if temporary_file_path.exists():
  45. temporary_file_path.remove()
  46. except Exception:
  47. pass
  48. raise e
  49. def handle_uploaded_file(user, uploaded_file):
  50. image = validate_uploaded_file(uploaded_file)
  51. store.store_temporary_avatar(user, image)
  52. def crop_string_to_dict(image, crop):
  53. message = _("Crop is invalid. Please try again.")
  54. crop_dict = {}
  55. try:
  56. crop_list = [int(x) for x in crop.split(',')]
  57. if len(crop_list) != 8:
  58. raise ValidationError(message)
  59. except TypeError, ValueError:
  60. raise ValidationError(message)
  61. cropped_size = (crop_list[0], crop_list[1])
  62. if cropped_size[0] < 10 or cropped_size[0] > image.size[0]:
  63. raise ValidationError(message)
  64. if cropped_size[1] < 10 or cropped_size[1] > image.size[1]:
  65. raise ValidationError(message)
  66. if crop_list[2] != crop_list[3]:
  67. # We only allow cropping to squares
  68. raise ValidationError(message)
  69. crop_dict['width'] = crop_list[2]
  70. crop_dict['height'] = crop_list[3]
  71. crop_dict['source'] = (crop_list[4], crop_list[6],
  72. crop_list[5], crop_list[7])
  73. if crop_dict['source'][0] < 0 or crop_dict['source'][2] > cropped_size[0]:
  74. raise ValidationError(message)
  75. if crop_dict['source'][1] < 0 or crop_dict['source'][3] > cropped_size[1]:
  76. raise ValidationError(message)
  77. source_w = crop_dict['source'][2] - crop_dict['source'][0]
  78. source_h = crop_dict['source'][3] - crop_dict['source'][1]
  79. if source_w != source_h:
  80. raise ValidationError(message)
  81. crop_dict['ratio'] = float(image.size[0]) / float(cropped_size[0])
  82. return crop_dict
  83. def crop_source_image(user, source, crop):
  84. image = Image.open(store.avatar_file_path(user, source))
  85. crop = crop_string_to_dict(image, crop)
  86. crop_dimensions = [int(d * crop['ratio']) for d in crop['source']]
  87. cropped_image = image.crop(crop_dimensions)
  88. store.store_avatar(user, cropped_image)
  89. if source == 'tmp':
  90. store.store_original_avatar(user)
  91. return crop
  92. def avatar_source_token(user, source):
  93. token_seed = (
  94. unicode(user.pk),
  95. user.username,
  96. user.email,
  97. source,
  98. unicode(store.avatar_file_path(user, source)),
  99. settings.SECRET_KEY
  100. )
  101. return sha256('+'.join(token_seed)).hexdigest()[:10]
  102. def has_temporary_avatar(user):
  103. return store.avatar_file_exists(user, 'tmp')
  104. def has_original_avatar(user):
  105. return store.avatar_file_exists(user, 'org')