photo-objects 0.6.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.6.0 → photo_objects-0.7.0}/PKG-INFO +1 -1
  2. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/api/photo.py +23 -5
  3. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/api/utils.py +6 -0
  4. photo_objects-0.7.0/photo_objects/django/context_processors.py +13 -0
  5. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/forms.py +5 -1
  6. photo_objects-0.7.0/photo_objects/django/tests/test_og_meta.py +60 -0
  7. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/tests/test_photo.py +9 -0
  8. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/tests/utils.py +21 -0
  9. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/views/ui/photo.py +14 -2
  10. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects.egg-info/PKG-INFO +1 -1
  11. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects.egg-info/SOURCES.txt +1 -0
  12. {photo_objects-0.6.0 → photo_objects-0.7.0}/pyproject.toml +1 -1
  13. photo_objects-0.6.0/photo_objects/django/context_processors.py +0 -9
  14. {photo_objects-0.6.0 → photo_objects-0.7.0}/LICENSE +0 -0
  15. {photo_objects-0.6.0 → photo_objects-0.7.0}/README.md +0 -0
  16. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/__init__.py +0 -0
  17. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/config.py +0 -0
  18. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/__init__.py +0 -0
  19. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/admin.py +0 -0
  20. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/api/__init__.py +0 -0
  21. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/api/album.py +0 -0
  22. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/api/auth.py +0 -0
  23. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/apps.py +0 -0
  24. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/conf.py +0 -0
  25. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/management/__init__.py +0 -0
  26. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/management/commands/__init__.py +0 -0
  27. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/management/commands/clean-scaled-photos.py +0 -0
  28. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/management/commands/create-initial-admin-account.py +0 -0
  29. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/migrations/0001_initial.py +0 -0
  30. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/migrations/0002_created_at_updated_at.py +0 -0
  31. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/migrations/0003_admin_visibility.py +0 -0
  32. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/migrations/0004_camera_setup_and_settings.py +0 -0
  33. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/migrations/0005_sitesettings.py +0 -0
  34. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/migrations/__init__.py +0 -0
  35. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/models.py +0 -0
  36. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/objsto.py +0 -0
  37. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/signals.py +0 -0
  38. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/templatetags/__init__.py +0 -0
  39. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/templatetags/photo_objects_extras.py +0 -0
  40. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/tests/__init__.py +0 -0
  41. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/tests/test_album.py +0 -0
  42. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/tests/test_auth.py +0 -0
  43. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/tests/test_commands.py +0 -0
  44. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/tests/test_img.py +0 -0
  45. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/tests/test_utils.py +0 -0
  46. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/urls.py +0 -0
  47. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/views/__init__.py +0 -0
  48. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/views/api/__init__.py +0 -0
  49. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/views/api/album.py +0 -0
  50. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/views/api/auth.py +0 -0
  51. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/views/api/photo.py +0 -0
  52. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/views/api/utils.py +0 -0
  53. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/views/ui/__init__.py +0 -0
  54. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/views/ui/album.py +0 -0
  55. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/views/ui/configuration.py +0 -0
  56. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/views/ui/users.py +0 -0
  57. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/views/ui/utils.py +0 -0
  58. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/django/views/utils.py +0 -0
  59. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/error.py +0 -0
  60. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/img.py +0 -0
  61. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects/utils.py +0 -0
  62. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects.egg-info/dependency_links.txt +0 -0
  63. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects.egg-info/requires.txt +0 -0
  64. {photo_objects-0.6.0 → photo_objects-0.7.0}/photo_objects.egg-info/top_level.txt +0 -0
  65. {photo_objects-0.6.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.6.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()
@@ -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
 
@@ -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
@@ -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,
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: photo-objects
3
- Version: 0.6.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.6.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