attachment.py 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  1. import os
  2. from hashlib import md5
  3. from io import BytesIO
  4. from PIL import Image
  5. from django.core.files import File
  6. from django.core.files.base import ContentFile
  7. from django.db import models
  8. from django.urls import reverse
  9. from django.utils import timezone
  10. from django.utils.crypto import get_random_string
  11. from misago.conf import settings
  12. from misago.core.utils import slugify
  13. def upload_to(instance, filename):
  14. spread_path = md5(str(instance.secret[:16]).encode()).hexdigest()
  15. secret = Attachment.generate_new_secret()
  16. filename_lowered = filename.lower().strip()
  17. for extension in instance.filetype.extensions_list:
  18. if filename_lowered.endswith(extension):
  19. break
  20. filename_clean = '.'.join((slugify(filename[:(len(extension) + 1) * -1])[:16], extension))
  21. return os.path.join('attachments', spread_path[:2], spread_path[2:4], secret, filename_clean)
  22. class Attachment(models.Model):
  23. secret = models.CharField(max_length=64)
  24. filetype = models.ForeignKey('AttachmentType', on_delete=models.CASCADE)
  25. post = models.ForeignKey('Post', blank=True, null=True, on_delete=models.SET_NULL)
  26. uploaded_on = models.DateTimeField(default=timezone.now, db_index=True)
  27. uploader = models.ForeignKey(
  28. settings.AUTH_USER_MODEL,
  29. blank=True,
  30. null=True,
  31. on_delete=models.SET_NULL,
  32. )
  33. uploader_name = models.CharField(max_length=255)
  34. uploader_slug = models.CharField(max_length=255, db_index=True)
  35. filename = models.CharField(max_length=255, db_index=True)
  36. size = models.PositiveIntegerField(default=0, db_index=True)
  37. thumbnail = models.ImageField(max_length=255, blank=True, null=True, upload_to=upload_to)
  38. image = models.ImageField(max_length=255, blank=True, null=True, upload_to=upload_to)
  39. file = models.FileField(max_length=255, blank=True, null=True, upload_to=upload_to)
  40. def __str__(self):
  41. return self.filename
  42. def delete(self, *args, **kwargs):
  43. self.delete_files()
  44. return super().delete(*args, **kwargs)
  45. def delete_files(self):
  46. if self.thumbnail:
  47. self.thumbnail.delete(save=False)
  48. if self.image:
  49. self.image.delete(save=False)
  50. if self.file:
  51. self.file.delete(save=False)
  52. @classmethod
  53. def generate_new_secret(cls):
  54. return get_random_string(settings.MISAGO_ATTACHMENT_SECRET_LENGTH)
  55. @property
  56. def is_image(self):
  57. return bool(self.image)
  58. @property
  59. def is_file(self):
  60. return not self.is_image
  61. def get_absolute_url(self):
  62. return reverse(
  63. 'misago:attachment', kwargs={
  64. 'pk': self.pk,
  65. 'secret': self.secret,
  66. }
  67. )
  68. def get_thumbnail_url(self):
  69. if self.thumbnail:
  70. return reverse(
  71. 'misago:attachment-thumbnail', kwargs={
  72. 'pk': self.pk,
  73. 'secret': self.secret,
  74. }
  75. )
  76. else:
  77. return None
  78. def set_file(self, upload):
  79. self.file = File(upload, upload.name)
  80. def set_image(self, upload):
  81. fileformat = self.filetype.extensions_list[0]
  82. self.image = File(upload, upload.name)
  83. thumbnail = Image.open(upload)
  84. downscale_image = (
  85. thumbnail.size[0] > settings.MISAGO_ATTACHMENT_IMAGE_SIZE_LIMIT[0]
  86. or thumbnail.size[1] > settings.MISAGO_ATTACHMENT_IMAGE_SIZE_LIMIT[1]
  87. )
  88. strip_animation = fileformat == 'gif'
  89. thumb_stream = BytesIO()
  90. if downscale_image:
  91. thumbnail.thumbnail(settings.MISAGO_ATTACHMENT_IMAGE_SIZE_LIMIT)
  92. if fileformat == 'jpg':
  93. # normalize jpg to jpeg for Pillow
  94. thumbnail.save(thumb_stream, 'jpeg')
  95. else:
  96. thumbnail.save(thumb_stream, fileformat)
  97. elif strip_animation:
  98. thumbnail.save(thumb_stream, fileformat)
  99. if downscale_image or strip_animation:
  100. self.thumbnail = ContentFile(thumb_stream.getvalue(), upload.name)