photo-objects 0.5.0__tar.gz → 0.7.0__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (65) hide show
  1. {photo_objects-0.5.0 → photo_objects-0.7.0}/PKG-INFO +1 -1
  2. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/api/photo.py +23 -5
  3. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/api/utils.py +6 -0
  4. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/conf.py +10 -5
  5. photo_objects-0.7.0/photo_objects/django/context_processors.py +13 -0
  6. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/forms.py +5 -1
  7. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/objsto.py +26 -2
  8. photo_objects-0.7.0/photo_objects/django/tests/test_og_meta.py +60 -0
  9. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/tests/test_photo.py +9 -0
  10. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/tests/utils.py +21 -0
  11. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/views/api/photo.py +13 -4
  12. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/views/ui/photo.py +14 -2
  13. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/img.py +7 -3
  14. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects.egg-info/PKG-INFO +1 -1
  15. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects.egg-info/SOURCES.txt +1 -0
  16. {photo_objects-0.5.0 → photo_objects-0.7.0}/pyproject.toml +1 -1
  17. photo_objects-0.5.0/photo_objects/django/context_processors.py +0 -9
  18. {photo_objects-0.5.0 → photo_objects-0.7.0}/LICENSE +0 -0
  19. {photo_objects-0.5.0 → photo_objects-0.7.0}/README.md +0 -0
  20. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/__init__.py +0 -0
  21. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/config.py +0 -0
  22. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/__init__.py +0 -0
  23. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/admin.py +0 -0
  24. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/api/__init__.py +0 -0
  25. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/api/album.py +0 -0
  26. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/api/auth.py +0 -0
  27. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/apps.py +0 -0
  28. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/management/__init__.py +0 -0
  29. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/management/commands/__init__.py +0 -0
  30. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/management/commands/clean-scaled-photos.py +0 -0
  31. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/management/commands/create-initial-admin-account.py +0 -0
  32. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/migrations/0001_initial.py +0 -0
  33. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/migrations/0002_created_at_updated_at.py +0 -0
  34. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/migrations/0003_admin_visibility.py +0 -0
  35. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/migrations/0004_camera_setup_and_settings.py +0 -0
  36. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/migrations/0005_sitesettings.py +0 -0
  37. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/migrations/__init__.py +0 -0
  38. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/models.py +0 -0
  39. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/signals.py +0 -0
  40. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/templatetags/__init__.py +0 -0
  41. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/templatetags/photo_objects_extras.py +0 -0
  42. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/tests/__init__.py +0 -0
  43. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/tests/test_album.py +0 -0
  44. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/tests/test_auth.py +0 -0
  45. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/tests/test_commands.py +0 -0
  46. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/tests/test_img.py +0 -0
  47. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/tests/test_utils.py +0 -0
  48. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/urls.py +0 -0
  49. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/views/__init__.py +0 -0
  50. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/views/api/__init__.py +0 -0
  51. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/views/api/album.py +0 -0
  52. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/views/api/auth.py +0 -0
  53. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/views/api/utils.py +0 -0
  54. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/views/ui/__init__.py +0 -0
  55. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/views/ui/album.py +0 -0
  56. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/views/ui/configuration.py +0 -0
  57. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/views/ui/users.py +0 -0
  58. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/views/ui/utils.py +0 -0
  59. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/django/views/utils.py +0 -0
  60. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/error.py +0 -0
  61. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects/utils.py +0 -0
  62. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects.egg-info/dependency_links.txt +0 -0
  63. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects.egg-info/requires.txt +0 -0
  64. {photo_objects-0.5.0 → photo_objects-0.7.0}/photo_objects.egg-info/top_level.txt +0 -0
  65. {photo_objects-0.5.0 → photo_objects-0.7.0}/setup.cfg +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: photo-objects
3
- Version: 0.5.0
3
+ Version: 0.7.0
4
4
  Summary: Application for storing photos in S3 compatible object-storage.
5
5
  Author: Toni Kangas
6
6
  License: MIT License
@@ -17,6 +17,7 @@ from photo_objects.img import photo_details
17
17
  from .auth import check_album_access, check_photo_access
18
18
  from .utils import (
19
19
  FormValidationFailed,
20
+ UploadPhotosFailed,
20
21
  JsonProblem,
21
22
  check_permissions,
22
23
  parse_input_data,
@@ -57,7 +58,7 @@ def _upload_photo(album_key: str, photo_file: UploadedFile):
57
58
  photo.delete()
58
59
 
59
60
  msg = objsto.with_error_code(
60
- "Could not save photo to object storage", e)
61
+ "Could not save photo to object storage.", e)
61
62
  logger.error(f"{msg}: {str(e)}")
62
63
  raise JsonProblem(f"{msg}.", 500) from e
63
64
 
@@ -73,6 +74,17 @@ def upload_photo(request: HttpRequest, album_key: str):
73
74
  return _upload_photo(album_key, photo_file)
74
75
 
75
76
 
77
+ def _join_errors(errors: dict) -> str:
78
+ messages = []
79
+ for _, errs in errors.items():
80
+ for err in errs:
81
+ try:
82
+ messages.append(err.get('message'))
83
+ except AttributeError:
84
+ messages.append(str(err))
85
+ return " ".join(messages) if messages else "Unknown error."
86
+
87
+
76
88
  def upload_photos(request: HttpRequest, album_key: str):
77
89
  check_permissions(
78
90
  request,
@@ -92,12 +104,18 @@ def upload_photos(request: HttpRequest, album_key: str):
92
104
  for photo_file in f.cleaned_data["photos"]:
93
105
  try:
94
106
  photos.append(_upload_photo(album_key, photo_file))
95
- except JsonProblem:
96
- # TODO: include error type in the message
97
- f.add_error("photos", f"Failed to upload {photo_file.name}.")
107
+ except FormValidationFailed as e:
108
+ message = _join_errors(e.form.errors.get_json_data())
109
+ f.add_error(
110
+ "photos",
111
+ f"Failed to upload {photo_file.name}. {message}")
112
+ except JsonProblem as e:
113
+ f.add_error(
114
+ "photos",
115
+ f"Failed to upload {photo_file.name}. {e.title}")
98
116
 
99
117
  if not f.is_valid():
100
- raise FormValidationFailed(f)
118
+ raise UploadPhotosFailed(f, [i.filename for i in photos])
101
119
 
102
120
  return photos
103
121
 
@@ -129,6 +129,12 @@ class FormValidationFailed(JsonProblem):
129
129
  self.form = form
130
130
 
131
131
 
132
+ class UploadPhotosFailed(FormValidationFailed):
133
+ def __init__(self, form: ModelForm, uploaded_photos: list[str]):
134
+ super().__init__(form)
135
+ self.uploaded_photos = uploaded_photos
136
+
137
+
132
138
  def check_permissions(request: HttpRequest, *permissions: str):
133
139
  if not request.user.is_authenticated:
134
140
  raise Unauthorized()
@@ -30,20 +30,25 @@ class PhotoSizeDimensions:
30
30
  max_width: int = None
31
31
  max_height: int = None
32
32
  max_aspect_ratio: float = None
33
+ image_format: str = None
33
34
 
34
35
 
35
36
  DEFAULT_SM = dict(
36
37
  max_width=512,
37
38
  max_height=512,
38
- max_aspect_ratio=1.5
39
+ max_aspect_ratio=1.5,
40
+ image_format='WEBP'
39
41
  )
40
42
  DEFAULT_MD = dict(
41
- max_width=2048,
42
- max_height=2048
43
+ max_width=1024,
44
+ max_height=1024,
45
+ max_aspect_ratio=1.5,
46
+ image_format="JPEG"
43
47
  )
44
48
  DEFAULT_LG = dict(
45
- max_width=4096,
46
- max_height=4096
49
+ max_width=2048,
50
+ max_height=2048,
51
+ image_format='WEBP'
47
52
  )
48
53
 
49
54
 
@@ -0,0 +1,13 @@
1
+ from importlib.metadata import PackageNotFoundError, version
2
+
3
+
4
+ class Metadata:
5
+ def __init__(self):
6
+ try:
7
+ self.version = version('photo_objects')
8
+ except PackageNotFoundError:
9
+ self.version = None
10
+
11
+
12
+ def metadata(_):
13
+ return {"photo_objects_metadata": Metadata()}
@@ -195,7 +195,11 @@ class CreatePhotoForm(ModelForm):
195
195
  error_messages = {
196
196
  'album': {
197
197
  'invalid_choice': _('Album with %(value)s key does not exist.')
198
- }
198
+ },
199
+ 'key': {
200
+ 'unique': _(
201
+ 'Photo with this filename already exists in the album.'),
202
+ },
199
203
  }
200
204
 
201
205
 
@@ -2,6 +2,7 @@ from dataclasses import asdict
2
2
  from io import BytesIO
3
3
  import json
4
4
  import mimetypes
5
+ import re
5
6
  import urllib3
6
7
 
7
8
  from minio import Minio, S3Error
@@ -60,8 +61,30 @@ def photo_path(album_key, photo_key, size_key):
60
61
  return f"{size_key}/{album_key}/{photo_key}"
61
62
 
62
63
 
63
- def put_photo(album_key, photo_key, size_key, photo_file):
64
- content_type = mimetypes.guess_type(photo_key)[0]
64
+ def _photo_filename(photo_key: str, image_format: str = None) -> str:
65
+ if image_format:
66
+ filename = re.sub(r'\.[^.]+$', '', photo_key)
67
+ return f"{filename}.{image_format.lower()}"
68
+
69
+ return photo_key
70
+
71
+
72
+ def photo_content_headers(
73
+ photo_key: str,
74
+ image_format: str = None,
75
+ ) -> tuple[str, dict[str, str]]:
76
+ filename = _photo_filename(photo_key, image_format)
77
+
78
+ content_type = mimetypes.guess_type(filename, strict=False)[0]
79
+ headers = {
80
+ "Content-Disposition": f"inline; filename={filename}"
81
+ }
82
+
83
+ return content_type, headers
84
+
85
+
86
+ def put_photo(album_key, photo_key, size_key, photo_file, image_format=None):
87
+ content_type, headers = photo_content_headers(photo_key, image_format)
65
88
 
66
89
  client, bucket = _objsto_access()
67
90
  return client.put_object(
@@ -71,6 +94,7 @@ def put_photo(album_key, photo_key, size_key, photo_file):
71
94
  length=-1,
72
95
  part_size=10 * MEGABYTE,
73
96
  content_type=content_type,
97
+ metadata=headers
74
98
  )
75
99
 
76
100
 
@@ -0,0 +1,60 @@
1
+ from django.contrib.sites.models import Site
2
+
3
+ from photo_objects.django.models import Album, Photo, SiteSettings
4
+
5
+ from .utils import TestCase, create_dummy_photo, temp_static_files
6
+
7
+
8
+ PHOTOS_DIRECTORY = "photos"
9
+
10
+
11
+ class OgMetaTests(TestCase):
12
+ @classmethod
13
+ def setUpTestData(cls):
14
+ album = Album.objects.create(
15
+ key="paris", visibility=Album.Visibility.PUBLIC)
16
+
17
+ create_dummy_photo(album, "tower.jpeg")
18
+
19
+ @temp_static_files
20
+ def test_albums_og_meta(self):
21
+ og_title = '<meta property="og:title" content="Test" />'
22
+
23
+ response = self.client.get("/albums")
24
+ self.assertNotContains(
25
+ response,
26
+ og_title,
27
+ status_code=200,
28
+ html=True)
29
+
30
+ site = Site.objects.get(id=1)
31
+ site.name = "Test"
32
+ site.domain = "test.example.com"
33
+ site.save()
34
+
35
+ response = self.client.get("/albums")
36
+ self.assertNotContains(
37
+ response,
38
+ og_title,
39
+ status_code=200,
40
+ html=True)
41
+
42
+ site_settings = SiteSettings.objects.get(site=site)
43
+ site_settings.description = "Description"
44
+ site_settings.preview_image = Photo.objects.get(key="paris/tower.jpeg")
45
+ site_settings.save()
46
+
47
+ tags = [
48
+ og_title,
49
+ '<meta property="og:description" content="Description" />',
50
+ '<meta property="og:image" content="https://test.example.com/img/paris/tower.jpeg/md"/>', # noqa: E501
51
+ '<meta property="og:url" content="https://test.example.com/albums" />', # noqa: E501
52
+ ]
53
+
54
+ response = self.client.get("/albums")
55
+ for tag in tags:
56
+ self.assertContains(
57
+ response,
58
+ tag,
59
+ status_code=200,
60
+ html=True)
@@ -184,6 +184,15 @@ class PhotoViewTests(TestCase):
184
184
  {filename: file})
185
185
  self.assertStatus(response, 500)
186
186
 
187
+ response = self.client.get(
188
+ "/api/albums/test-photo-a/photos/tower.jpg")
189
+ self.assertStatus(response, 404)
190
+
191
+ response = self.client.get(
192
+ "/api/albums/test-photo-a/photos")
193
+ self.assertStatus(response, 200)
194
+ self.assertEqual(len(response.json()), 0)
195
+
187
196
  def test_get_image_scales_the_image(self):
188
197
  login_success = self.client.login(
189
198
  username='has_permission', password='test')
@@ -1,7 +1,11 @@
1
1
  from dataclasses import dataclass
2
+ from io import StringIO
2
3
  import os
4
+ import shutil
5
+ import tempfile
3
6
 
4
7
  from django.conf import settings
8
+ from django.core.management import call_command
5
9
  from django.test import TestCase as DjangoTestCase, override_settings
6
10
  from django.utils import timezone
7
11
  from django.utils.dateparse import parse_datetime
@@ -120,3 +124,20 @@ def parse_timestamps(data):
120
124
  return Timestamps(
121
125
  created_at=data.get('created_at'),
122
126
  updated_at=data.get('updated_at'))
127
+
128
+
129
+ # Based on https://stackoverflow.com/a/76745063
130
+ def temp_static_files(func):
131
+ '''Decorator that creates a temporary directory, configures that as
132
+ STATIC_ROOT, and collects static files there.'''
133
+ def wrapper(*args, **kwargs):
134
+ static_root = tempfile.mkdtemp(prefix="test_static_")
135
+ with override_settings(STATIC_ROOT=static_root):
136
+ try:
137
+ out = StringIO()
138
+ call_command("collectstatic", "--noinput", stdout=out)
139
+ func(*args, **kwargs)
140
+ finally:
141
+ shutil.rmtree(static_root)
142
+
143
+ return wrapper
@@ -91,14 +91,23 @@ def get_img(request: HttpRequest, album_key: str, photo_key: str):
91
91
  404 if code == "NoSuchKey" else 500,
92
92
  ).json_response
93
93
 
94
- sizes = photo_sizes()
94
+ size_params = getattr(photo_sizes(), size)
95
95
  # TODO: handle error
96
96
  scaled_photo = scale_photo(
97
- original_photo, photo_key, **asdict(getattr(sizes, size)))
97
+ original_photo, photo_key, **asdict(size_params))
98
98
 
99
99
  # TODO: handle error
100
100
  scaled_photo.seek(0)
101
- objsto.put_photo(album_key, photo_key, size, scaled_photo)
101
+ objsto.put_photo(
102
+ album_key,
103
+ photo_key,
104
+ size,
105
+ scaled_photo,
106
+ size_params.image_format)
107
+
108
+ content_type, headers = objsto.photo_content_headers(
109
+ photo_key, size_params.image_format)
102
110
 
103
111
  scaled_photo.seek(0)
104
- return HttpResponse(scaled_photo.read(), content_type=content_type)
112
+ return HttpResponse(
113
+ scaled_photo.read(), content_type=content_type, headers=headers)
@@ -3,7 +3,11 @@ from django.shortcuts import render
3
3
  from django.urls import reverse
4
4
 
5
5
  from photo_objects.django import api
6
- from photo_objects.django.api.utils import AlbumNotFound, FormValidationFailed
6
+ from photo_objects.django.api.utils import (
7
+ AlbumNotFound,
8
+ FormValidationFailed,
9
+ UploadPhotosFailed,
10
+ )
7
11
  from photo_objects.django.forms import ModifyPhotoForm, UploadPhotosForm
8
12
  from photo_objects.django.models import Photo
9
13
  from photo_objects.django.views.utils import (
@@ -24,10 +28,17 @@ def upload_photos(request: HttpRequest, album_key: str):
24
28
  reverse(
25
29
  'photo_objects:show_album',
26
30
  kwargs={"album_key": album_key}))
27
- except FormValidationFailed as e:
31
+ except UploadPhotosFailed as e:
28
32
  form = e.form
33
+ uploaded_count = len(e.uploaded_photos)
34
+ if uploaded_count > 0:
35
+ plural = 's' if uploaded_count != 1 else ''
36
+ info = f"Successfully uploaded {uploaded_count} photo{plural}."
37
+ else:
38
+ info = None
29
39
  else:
30
40
  form = UploadPhotosForm()
41
+ info = None
31
42
 
32
43
  album = api.check_album_access(request, album_key)
33
44
  target = album.title or album.key
@@ -38,6 +49,7 @@ def upload_photos(request: HttpRequest, album_key: str):
38
49
 
39
50
  return render(request, 'photo_objects/photo/upload.html', {
40
51
  "form": form,
52
+ "info": info,
41
53
  "title": "Upload photos",
42
54
  "back": back,
43
55
  "photo": album.cover_photo,
@@ -66,7 +66,10 @@ def _read_camera_setup_and_settings(image: Image) -> dict:
66
66
  raise e
67
67
 
68
68
 
69
- def _image_format(filename):
69
+ def _image_format(image_format, filename):
70
+ if image_format:
71
+ return image_format
72
+
70
73
  image_format = filename.split('.')[-1].upper()
71
74
 
72
75
  if image_format == "JPG":
@@ -131,7 +134,8 @@ def scale_photo(
131
134
  filename,
132
135
  max_width=None,
133
136
  max_height=None,
134
- max_aspect_ratio=None):
137
+ max_aspect_ratio=None,
138
+ image_format=None):
135
139
  image = Image.open(photo_file)
136
140
  width, height = image.size
137
141
 
@@ -164,5 +168,5 @@ def scale_photo(
164
168
  resized = image.resize(new_size, Image.Resampling.LANCZOS)
165
169
 
166
170
  b = BytesIO()
167
- resized.save(b, _image_format(filename))
171
+ resized.save(b, _image_format(image_format, filename))
168
172
  return b
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: photo-objects
3
- Version: 0.5.0
3
+ Version: 0.7.0
4
4
  Summary: Application for storing photos in S3 compatible object-storage.
5
5
  Author: Toni Kangas
6
6
  License: MIT License
@@ -43,6 +43,7 @@ photo_objects/django/tests/test_album.py
43
43
  photo_objects/django/tests/test_auth.py
44
44
  photo_objects/django/tests/test_commands.py
45
45
  photo_objects/django/tests/test_img.py
46
+ photo_objects/django/tests/test_og_meta.py
46
47
  photo_objects/django/tests/test_photo.py
47
48
  photo_objects/django/tests/test_utils.py
48
49
  photo_objects/django/tests/utils.py
@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
4
4
 
5
5
  [project]
6
6
  name = "photo-objects"
7
- version = "0.5.0"
7
+ version = "0.7.0"
8
8
  dependencies = [
9
9
  "markdown~=3.7",
10
10
  "minio~=7.2",
@@ -1,9 +0,0 @@
1
- from importlib.metadata import version
2
-
3
-
4
- class Metadata:
5
- version = version('photo_objects')
6
-
7
-
8
- def metadata(_):
9
- return {"photo_objects_metadata": Metadata()}
File without changes
File without changes
File without changes