attachment.py 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. import os
  2. from hashlib import md5
  3. from io import BytesIO
  4. from django.core.files import File
  5. from django.core.files.base import ContentFile
  6. from django.db import models
  7. from django.urls import reverse
  8. from django.utils import timezone
  9. from django.utils.crypto import get_random_string
  10. from PIL import Image
  11. from ...conf import settings
  12. from ...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(
  21. (slugify(filename[: (len(extension) + 1) * -1])[:16], extension)
  22. )
  23. return os.path.join(
  24. "attachments", spread_path[:2], spread_path[2:4], secret, filename_clean
  25. )
  26. class Attachment(models.Model):
  27. secret = models.CharField(max_length=64)
  28. filetype = models.ForeignKey("AttachmentType", on_delete=models.CASCADE)
  29. post = models.ForeignKey("Post", blank=True, null=True, on_delete=models.SET_NULL)
  30. uploaded_on = models.DateTimeField(default=timezone.now, db_index=True)
  31. uploader = models.ForeignKey(
  32. settings.AUTH_USER_MODEL, blank=True, null=True, on_delete=models.SET_NULL
  33. )
  34. uploader_name = models.CharField(max_length=255)
  35. uploader_slug = models.CharField(max_length=255, db_index=True)
  36. filename = models.CharField(max_length=255, db_index=True)
  37. size = models.PositiveIntegerField(default=0, db_index=True)
  38. thumbnail = models.ImageField(
  39. max_length=255, blank=True, null=True, upload_to=upload_to
  40. )
  41. image = models.ImageField(
  42. max_length=255, blank=True, null=True, upload_to=upload_to
  43. )
  44. file = models.FileField(max_length=255, blank=True, null=True, upload_to=upload_to)
  45. def __str__(self):
  46. return self.filename
  47. def delete(self, *args, **kwargs):
  48. self.delete_files()
  49. return super().delete(*args, **kwargs)
  50. def delete_files(self):
  51. if self.thumbnail:
  52. self.thumbnail.delete(save=False)
  53. if self.image:
  54. self.image.delete(save=False)
  55. if self.file:
  56. self.file.delete(save=False)
  57. @classmethod
  58. def generate_new_secret(cls):
  59. return get_random_string(settings.MISAGO_ATTACHMENT_SECRET_LENGTH)
  60. @property
  61. def is_image(self):
  62. return bool(self.image)
  63. @property
  64. def is_file(self):
  65. return not self.is_image
  66. def get_absolute_url(self):
  67. return reverse(
  68. "misago:attachment", kwargs={"pk": self.pk, "secret": self.secret}
  69. )
  70. def get_thumbnail_url(self):
  71. if self.thumbnail:
  72. return reverse(
  73. "misago:attachment-thumbnail",
  74. kwargs={"pk": self.pk, "secret": self.secret},
  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)