photo-objects 0.6.0__py3-none-any.whl → 0.7.1__py3-none-any.whl
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.
- photo_objects/django/api/photo.py +23 -5
- photo_objects/django/api/utils.py +6 -0
- photo_objects/django/context_processors.py +6 -2
- photo_objects/django/forms.py +5 -1
- photo_objects/django/tests/test_og_meta.py +60 -0
- photo_objects/django/tests/test_photo.py +9 -0
- photo_objects/django/tests/utils.py +21 -0
- photo_objects/django/views/ui/photo.py +14 -2
- {photo_objects-0.6.0.dist-info → photo_objects-0.7.1.dist-info}/METADATA +1 -1
- {photo_objects-0.6.0.dist-info → photo_objects-0.7.1.dist-info}/RECORD +13 -12
- {photo_objects-0.6.0.dist-info → photo_objects-0.7.1.dist-info}/WHEEL +0 -0
- {photo_objects-0.6.0.dist-info → photo_objects-0.7.1.dist-info}/licenses/LICENSE +0 -0
- {photo_objects-0.6.0.dist-info → photo_objects-0.7.1.dist-info}/top_level.txt +0 -0
|
@@ -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
|
|
96
|
-
|
|
97
|
-
f.add_error(
|
|
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
|
|
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()
|
|
@@ -1,8 +1,12 @@
|
|
|
1
|
-
from importlib.metadata import version
|
|
1
|
+
from importlib.metadata import PackageNotFoundError, version
|
|
2
2
|
|
|
3
3
|
|
|
4
4
|
class Metadata:
|
|
5
|
-
|
|
5
|
+
def __init__(self):
|
|
6
|
+
try:
|
|
7
|
+
self.version = version('photo_objects')
|
|
8
|
+
except PackageNotFoundError:
|
|
9
|
+
self.version = None
|
|
6
10
|
|
|
7
11
|
|
|
8
12
|
def metadata(_):
|
photo_objects/django/forms.py
CHANGED
|
@@ -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
|
|
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
|
|
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,
|
|
@@ -7,8 +7,8 @@ photo_objects/django/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hS
|
|
|
7
7
|
photo_objects/django/admin.py,sha256=ubYfhsWQ85_2moYGZCVneLQ2DVEetbVpvBNisIrtuhw,170
|
|
8
8
|
photo_objects/django/apps.py,sha256=Apqu6o6fpoxda18NQgKupvQRvTAZxVviIK_-dUR3rck,1444
|
|
9
9
|
photo_objects/django/conf.py,sha256=ZpeIulEc1tpr8AO52meNKOF30Xf5osbDtDyHvQRtkx4,2593
|
|
10
|
-
photo_objects/django/context_processors.py,sha256=
|
|
11
|
-
photo_objects/django/forms.py,sha256=
|
|
10
|
+
photo_objects/django/context_processors.py,sha256=XacUmcYV-4NMMMNXPWrHKdvNd6lfyamisngaVerREiU,306
|
|
11
|
+
photo_objects/django/forms.py,sha256=LcznhXrqPfqRY5b2SwHBbC9-uux1FiZFdQBlHsKX5NU,6649
|
|
12
12
|
photo_objects/django/models.py,sha256=M40ZFSIX3WgyVXfHQY9SXQOY0s3fpFqfCxNV3CBD5M8,5753
|
|
13
13
|
photo_objects/django/objsto.py,sha256=B7DxPWuqFaPFXPLhsHCFlqIzYl7EXLxcHde6zJDe89A,4238
|
|
14
14
|
photo_objects/django/signals.py,sha256=Q_Swjl_9z6B6zP-97D_ep5zGSAEgmQfwUz0utMDY93A,1624
|
|
@@ -16,8 +16,8 @@ photo_objects/django/urls.py,sha256=pCUTxg4xSd3ZD8BZVjULb9QpJQ03Ek6-sKoVnYG3-OY,
|
|
|
16
16
|
photo_objects/django/api/__init__.py,sha256=BnEHlm3mwyBy-1xhk-NFasgZa4fjCDjtfkBUoH0puPY,62
|
|
17
17
|
photo_objects/django/api/album.py,sha256=CJDeGLCuYoxGUDcjssZRpFnToxG_KVE9Ii7NduFW2ks,2003
|
|
18
18
|
photo_objects/django/api/auth.py,sha256=lS0S1tMVH2uN30g4jlixklv3eMnQ2FbQVQvuRXeMGYo,1420
|
|
19
|
-
photo_objects/django/api/photo.py,sha256
|
|
20
|
-
photo_objects/django/api/utils.py,sha256=
|
|
19
|
+
photo_objects/django/api/photo.py,sha256=-lo1g6jfBr884wy-WV2DAEPxzH9V-tFUTRtitmA6i28,4471
|
|
20
|
+
photo_objects/django/api/utils.py,sha256=xilnQndd1DXkNs6K6OqXpzUKkDSZFOG25xX3O2oJggU,5364
|
|
21
21
|
photo_objects/django/management/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
|
22
22
|
photo_objects/django/management/commands/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
|
23
23
|
photo_objects/django/management/commands/clean-scaled-photos.py,sha256=KJY6phgTCxcmbMUsUfCRQjatvCmKyFninM8zT-tB3Kc,2008
|
|
@@ -35,9 +35,10 @@ photo_objects/django/tests/test_album.py,sha256=yjxP_M0bddS9Xpg1d1Wk5OyJsmxmkdYu
|
|
|
35
35
|
photo_objects/django/tests/test_auth.py,sha256=hgr1UMVLvSI1x5zY7wTEXSBKfM5E_sNMIFlx8mVWYPY,3928
|
|
36
36
|
photo_objects/django/tests/test_commands.py,sha256=e3lE1ZhFR39WIq2VSKDNcQHUkSJqSWDYuAcAfu29svs,2955
|
|
37
37
|
photo_objects/django/tests/test_img.py,sha256=HEAWcr5fpTkzePkhoQ4YrWsDO9TvFOr7my_0LqVbaO4,829
|
|
38
|
-
photo_objects/django/tests/
|
|
38
|
+
photo_objects/django/tests/test_og_meta.py,sha256=Kk5a9KvE88KZ60gLqXSe6rTz5YU-gdjteksYolHd-nw,1804
|
|
39
|
+
photo_objects/django/tests/test_photo.py,sha256=aZCZw7PIXIfIYX3q1lUxn8C53mLRBNK0qp18t7X4ZQ4,13696
|
|
39
40
|
photo_objects/django/tests/test_utils.py,sha256=zBLv8lkvcLMCaH4D6GR1KZqUe-rPowhcBkQX19-Kshs,2007
|
|
40
|
-
photo_objects/django/tests/utils.py,sha256=
|
|
41
|
+
photo_objects/django/tests/utils.py,sha256=s_3hzQEY4tdja_8406s_SQyRC6fCYI_MBPqvFjC8fB4,4345
|
|
41
42
|
photo_objects/django/views/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
|
42
43
|
photo_objects/django/views/utils.py,sha256=oP9B6QtHPaukX97rqIlCHxanduktPGFVSCGkGxf4lzI,1691
|
|
43
44
|
photo_objects/django/views/api/__init__.py,sha256=SxK-b7MgMtD9VRMz46rDV5qtm6AvkRcg3Moa1AWl5pY,108
|
|
@@ -48,11 +49,11 @@ photo_objects/django/views/api/utils.py,sha256=uQzKdSKHRAux5OZzqgWQr0gsK_FeweQP0
|
|
|
48
49
|
photo_objects/django/views/ui/__init__.py,sha256=N3ro5KggdV-JnfyHwoStX73b3SbVbpcsMuQNlxntVJs,92
|
|
49
50
|
photo_objects/django/views/ui/album.py,sha256=PmVXAmqVjKLAie1NyB-qXO3eLqOmhIA8PTAGJewgxko,4738
|
|
50
51
|
photo_objects/django/views/ui/configuration.py,sha256=oX6SV0TFBpbaxfp4cXIdSL41YJhy_aOy30TkBxOpq0M,5065
|
|
51
|
-
photo_objects/django/views/ui/photo.py,sha256=
|
|
52
|
+
photo_objects/django/views/ui/photo.py,sha256=fVpKI-XqSV8KfoTpDYxceCHR3rJzuVYfmRRQx1I0YIQ,6649
|
|
52
53
|
photo_objects/django/views/ui/users.py,sha256=nb73cnzuV98QkJb0j8F2hqPgOGFIWpUFTFu6dXMeVwM,722
|
|
53
54
|
photo_objects/django/views/ui/utils.py,sha256=YV_YcUbX-zUkdFnBlezPChR6aPDhZJ9loSOHBSzF6Cc,273
|
|
54
|
-
photo_objects-0.
|
|
55
|
-
photo_objects-0.
|
|
56
|
-
photo_objects-0.
|
|
57
|
-
photo_objects-0.
|
|
58
|
-
photo_objects-0.
|
|
55
|
+
photo_objects-0.7.1.dist-info/licenses/LICENSE,sha256=V3w6hTjXfP65F4r_mejveHcV5igHrblxao3-2RlfVlA,1068
|
|
56
|
+
photo_objects-0.7.1.dist-info/METADATA,sha256=JfylWVQAoc7GE1HIjjBt9CXuvrS9tn6qr6_2eRodPfs,3605
|
|
57
|
+
photo_objects-0.7.1.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
|
|
58
|
+
photo_objects-0.7.1.dist-info/top_level.txt,sha256=SZeL8mhf-WMGdhRtTGFvZc3aIRBboQls9O0cFDMGdQ0,14
|
|
59
|
+
photo_objects-0.7.1.dist-info/RECORD,,
|
|
File without changes
|
|
File without changes
|
|
File without changes
|