test_avatars.py 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. from path import Path
  2. from PIL import Image
  3. from django.contrib.auth import get_user_model
  4. from django.core.exceptions import ValidationError
  5. from django.test import TestCase, override_settings
  6. from django.utils.crypto import get_random_string
  7. from misago.conf import settings
  8. from misago.users.avatars import (
  9. set_default_avatar, dynamic, gallery, gravatar, store, uploaded)
  10. from misago.users.models import Avatar, AvatarGallery
  11. UserModel = get_user_model()
  12. class AvatarsStoreTests(TestCase):
  13. def test_store(self):
  14. """store successfully stores and deletes avatar"""
  15. user = UserModel.objects.create_user(
  16. 'Bob', 'bob@bob.com', 'pass123')
  17. test_image = Image.new("RGBA", (100, 100), 0)
  18. store.store_new_avatar(user, test_image)
  19. # reload user
  20. test_user = UserModel.objects.get(pk=user.pk)
  21. # assert that avatars were stored in media
  22. avatars_dict = {}
  23. for avatar in user.avatar_set.all():
  24. self.assertTrue(avatar.image.url)
  25. self.assertEqual(avatar.url, avatar.image.url)
  26. avatars_dict[avatar.size] = avatar
  27. # asserts that user.avatars cache was set
  28. self.assertEqual(len(avatars_dict), len(settings.MISAGO_AVATARS_SIZES))
  29. self.assertEqual(len(user.avatars), len(settings.MISAGO_AVATARS_SIZES))
  30. self.assertEqual(len(user.avatars), len(avatars_dict))
  31. for avatar in user.avatars:
  32. self.assertIn(avatar['size'], settings.MISAGO_AVATARS_SIZES)
  33. self.assertEqual(avatar['url'], avatars_dict[avatar['size']].url)
  34. # another avatar change deleted old avatars
  35. store.store_new_avatar(user, test_image)
  36. for old_avatar in avatars_dict.values():
  37. avatar_path = Path(old_avatar.image.path)
  38. self.assertFalse(avatar_path.exists())
  39. self.assertFalse(avatar_path.isfile())
  40. with self.assertRaises(Avatar.DoesNotExist):
  41. Avatar.objects.get(pk=old_avatar.pk)
  42. # and updated user avatars again
  43. new_avatars_dict = {}
  44. for size in settings.MISAGO_AVATARS_SIZES:
  45. avatar = user.avatar_set.get(size=size)
  46. self.assertTrue(avatar.image.url)
  47. self.assertEqual(avatar.url, avatar.image.url)
  48. new_avatars_dict[size] = avatar
  49. self.assertTrue(avatars_dict != new_avatars_dict)
  50. # asserts that user.avatars cache was updated
  51. self.assertEqual(len(user.avatars), len(settings.MISAGO_AVATARS_SIZES))
  52. for avatar in user.avatars:
  53. self.assertIn(avatar['size'], settings.MISAGO_AVATARS_SIZES)
  54. self.assertEqual(avatar['url'], new_avatars_dict[avatar['size']].url)
  55. # delete avatar
  56. store.delete_avatar(user)
  57. for removed_avatar in new_avatars_dict.values():
  58. avatar_path = Path(removed_avatar.image.path)
  59. self.assertFalse(avatar_path.exists())
  60. self.assertFalse(avatar_path.isfile())
  61. with self.assertRaises(Avatar.DoesNotExist):
  62. Avatar.objects.get(pk=removed_avatar.pk)
  63. class AvatarSetterTests(TestCase):
  64. def setUp(self):
  65. self.user = UserModel.objects.create_user(
  66. 'Bob', 'kontakt@rpiton.com', 'pass123')
  67. self.user.avatars = None
  68. self.user.save()
  69. def tearDown(self):
  70. store.delete_avatar(self.user)
  71. def get_current_user(self):
  72. return UserModel.objects.get(pk=self.user.pk)
  73. def assertNoAvatarIsSet(self):
  74. user = self.get_current_user()
  75. self.assertFalse(user.avatars)
  76. def assertAvatarWasSet(self):
  77. user = self.get_current_user()
  78. avatars_dict = {}
  79. for avatar in user.avatar_set.all():
  80. avatar_path = Path(avatar.image.path)
  81. self.assertTrue(avatar_path.exists())
  82. self.assertTrue(avatar_path.isfile())
  83. avatars_dict[avatar.size] = avatar
  84. self.assertEqual(len(user.avatars), len(avatars_dict))
  85. self.assertEqual(len(user.avatars), len(settings.MISAGO_AVATARS_SIZES))
  86. def test_dynamic_avatar(self):
  87. """dynamic avatar gets created"""
  88. self.assertNoAvatarIsSet()
  89. dynamic.set_avatar(self.user)
  90. self.assertAvatarWasSet()
  91. def test_random_gallery_avatar_no_gallery(self):
  92. """runtime error is raised when no gallery exists"""
  93. with self.assertRaises(RuntimeError):
  94. gallery.set_random_avatar(self.user)
  95. def test_random_gallery_avatar(self):
  96. """dynamic avatar gets created"""
  97. gallery.load_avatar_galleries()
  98. self.assertNoAvatarIsSet()
  99. gallery.set_random_avatar(self.user)
  100. self.assertAvatarWasSet()
  101. def test_selected_gallery_avatar(self):
  102. """dynamic avatar gets created"""
  103. gallery.load_avatar_galleries()
  104. self.assertNoAvatarIsSet()
  105. test_avatar = AvatarGallery.objects.order_by('id').last()
  106. gallery.set_avatar(self.user, test_avatar)
  107. self.assertAvatarWasSet()
  108. def test_gravatar(self):
  109. """dynamic avatar gets created"""
  110. self.assertNoAvatarIsSet()
  111. gravatar.set_avatar(self.user)
  112. self.assertAvatarWasSet()
  113. def test_default_avatar_gravatar(self):
  114. """default gravatar gets set"""
  115. self.assertNoAvatarIsSet()
  116. set_default_avatar(self.user, 'gravatar', 'dynamic')
  117. self.assertAvatarWasSet()
  118. def test_default_avatar_gravatar_fallback_dynamic(self):
  119. """default gravatar fails but fallback dynamic works"""
  120. gibberish_email = '%s@%s.%s' % (
  121. get_random_string(6), get_random_string(6), get_random_string(3))
  122. self.user.set_email(gibberish_email)
  123. self.user.save()
  124. self.assertNoAvatarIsSet()
  125. set_default_avatar(self.user, 'gravatar', 'dynamic')
  126. self.assertAvatarWasSet()
  127. def test_default_avatar_gravatar_fallback_empty_gallery(self):
  128. """default both gravatar and fallback fail set"""
  129. gibberish_email = '%s@%s.%s' % (
  130. get_random_string(6), get_random_string(6), get_random_string(3))
  131. self.user.set_email(gibberish_email)
  132. self.user.save()
  133. self.assertNoAvatarIsSet()
  134. self.user.save()
  135. set_default_avatar(self.user, 'gravatar', 'gallery')
  136. self.assertAvatarWasSet()
  137. class MockAvatarFile(object):
  138. def __init__(self, size=None, name=None, mime=None):
  139. self.size = size
  140. self.name = name
  141. self.content_type = mime
  142. class UploadedAvatarTests(TestCase):
  143. def test_clean_crop(self):
  144. """crop validation and cleaning"""
  145. image = Image.new("RGBA", (200, 200), 0)
  146. with self.assertRaises(ValidationError):
  147. uploaded.clean_crop(image, "abc")
  148. with self.assertRaises(ValidationError):
  149. uploaded.clean_crop(image, {})
  150. with self.assertRaises(ValidationError):
  151. uploaded.clean_crop(image, {'offset': {'x': 'ugabuga'}})
  152. with self.assertRaises(ValidationError):
  153. uploaded.clean_crop(image, {
  154. 'offset': {
  155. 'x': 0,
  156. 'y': 0,
  157. },
  158. 'zoom': -2
  159. })
  160. with self.assertRaises(ValidationError):
  161. uploaded.clean_crop(image, {
  162. 'offset': {
  163. 'x': 0,
  164. 'y': 0,
  165. },
  166. 'zoom': 2
  167. })
  168. def test_uploaded_image_size_validation(self):
  169. """uploaded image size is validated"""
  170. image = MockAvatarFile(size=settings.avatar_upload_limit * 2024)
  171. with self.assertRaises(ValidationError):
  172. uploaded.validate_file_size(image)
  173. image = MockAvatarFile(size=settings.avatar_upload_limit * 1000)
  174. uploaded.validate_file_size(image)
  175. def test_uploaded_image_extension_validation(self):
  176. """uploaded image extension is validated"""
  177. for invalid_extension in ('.txt', '.zip', '.py', '.tiff'):
  178. with self.assertRaises(ValidationError):
  179. image = MockAvatarFile(name='test%s' % invalid_extension)
  180. uploaded.validate_extension(image)
  181. for valid_extension in uploaded.ALLOWED_EXTENSIONS:
  182. image = MockAvatarFile(name='test%s' % valid_extension)
  183. uploaded.validate_extension(image)
  184. def test_uploaded_image_mime_validation(self):
  185. """uploaded image mime type is validated"""
  186. image = MockAvatarFile(mime='fake/mime')
  187. with self.assertRaises(ValidationError):
  188. uploaded.validate_mime(image)
  189. for valid_mime in uploaded.ALLOWED_MIME_TYPES:
  190. image = MockAvatarFile(mime=valid_mime)
  191. uploaded.validate_mime(image)