Skip to content

Commit

Permalink
Optimize images (#97)
Browse files Browse the repository at this point in the history
* Add optimized_path field to attachment model

* Add image optimizer base class

* Exclude base image optimizer from code coverage

* Support downloading optimized attachments
  • Loading branch information
4c0n authored Nov 7, 2024
1 parent 0a68b74 commit 81ca5b5
Show file tree
Hide file tree
Showing 4 changed files with 54 additions and 7 deletions.
18 changes: 16 additions & 2 deletions meldingen_core/actions/attachment.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
from collections.abc import Collection
from enum import StrEnum
from typing import AsyncIterator, Generic, TypeVar
from uuid import uuid4

Expand Down Expand Up @@ -71,6 +72,11 @@ async def __call__(
return attachment


class AttachmentTypes(StrEnum):
ORIGINAL = "original"
OPTIMIZED = "optimized"


class DownloadAttachmentAction(Generic[A, A_co, M, M_co]):
_verify_token: TokenVerifier[M, M_co]
_attachment_repository: BaseAttachmentRepository[A, A_co]
Expand All @@ -86,7 +92,9 @@ def __init__(
self._attachment_repository = attachment_repository
self._filesystem = filesystem

async def __call__(self, melding_id: int, attachment_id: int, token: str) -> AsyncIterator[bytes]:
async def __call__(
self, melding_id: int, attachment_id: int, token: str, _type: AttachmentTypes
) -> AsyncIterator[bytes]:
melding = await self._verify_token(melding_id, token)

attachment = await self._attachment_repository.retrieve(attachment_id)
Expand All @@ -96,8 +104,14 @@ async def __call__(self, melding_id: int, attachment_id: int, token: str) -> Asy
if attachment.melding != melding:
raise NotFoundException(f"Melding with id {melding_id} does not have attachment with id {attachment_id}")

file_path = attachment.file_path
if _type == AttachmentTypes.OPTIMIZED:
if attachment.optimized_path is None:
raise NotFoundException("Optimized file not found")
file_path = attachment.optimized_path

try:
file = await self._filesystem.get_file(attachment.file_path)
file = await self._filesystem.get_file(file_path)
return await file.get_iterator()
except filesystem.NotFoundException as exception:
raise NotFoundException("File not found") from exception
Expand Down
6 changes: 6 additions & 0 deletions meldingen_core/image.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
from abc import ABCMeta, abstractmethod # pragma: no cover


class BaseImageOptimizer(metaclass=ABCMeta): # pragma: no cover
@abstractmethod
async def __call__(self, image_path: str) -> str: ...
1 change: 1 addition & 0 deletions meldingen_core/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -53,3 +53,4 @@ class Attachment:
file_path: str = field(init=False)
original_filename: str
melding: Melding
optimized_path: str | None = None
36 changes: 31 additions & 5 deletions tests/test_actions/test_attachment.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
from plugfs.filesystem import File, Filesystem

from meldingen_core.actions.attachment import (
AttachmentTypes,
DeleteAttachmentAction,
DownloadAttachmentAction,
ListAttachmentsAction,
Expand Down Expand Up @@ -67,7 +68,7 @@ async def test_attachment_not_found(self) -> None:
)

with pytest.raises(NotFoundException) as exception_info:
await action(123, 456, "supersecrettoken")
await action(123, 456, "supersecrettoken", AttachmentTypes.ORIGINAL)

assert str(exception_info.value) == "Attachment not found"

Expand All @@ -85,18 +86,20 @@ async def test_attachment_does_not_belong_to_melding(self) -> None:
)

with pytest.raises(NotFoundException) as exception_info:
await action(123, 456, "supersecrettoken")
await action(123, 456, "supersecrettoken", AttachmentTypes.ORIGINAL)

assert str(exception_info.value) == "Melding with id 123 does not have attachment with id 456"

@pytest.mark.anyio
async def test_can_handle_attachment_download(self) -> None:
@pytest.mark.parametrize("_type", AttachmentTypes)
async def test_can_handle_attachment_download(self, _type: AttachmentTypes) -> None:
melding = Melding(text="text")
token_verifier = AsyncMock(TokenVerifier)
token_verifier.return_value = melding

attachment = Attachment("bla", melding)
attachment.file_path = "/path/to/file.ext"
attachment.optimized_path = "/path/to/file-optimized.ext"

attachment_repository = Mock(BaseAttachmentRepository)
attachment_repository.retrieve.return_value = attachment
Expand All @@ -107,7 +110,30 @@ async def test_can_handle_attachment_download(self) -> None:
Mock(Filesystem),
)

await action(123, 456, "supersecrettoken")
await action(123, 456, "supersecrettoken", _type)

@pytest.mark.anyio
async def test_optimized_path_none(self) -> None:
melding = Melding(text="text")
token_verifier = AsyncMock(TokenVerifier)
token_verifier.return_value = melding

attachment = Attachment("bla", melding)
attachment.file_path = "/path/to/file.ext"

attachment_repository = Mock(BaseAttachmentRepository)
attachment_repository.retrieve.return_value = attachment

action: DownloadAttachmentAction[Attachment, Attachment, Melding, Melding] = DownloadAttachmentAction(
token_verifier,
attachment_repository,
Mock(Filesystem),
)

with pytest.raises(NotFoundException) as exception_info:
await action(123, 456, "supersecrettoken", AttachmentTypes.OPTIMIZED)

assert str(exception_info.value) == "Optimized file not found"

@pytest.mark.anyio
async def test_file_not_found(self) -> None:
Expand All @@ -134,7 +160,7 @@ async def test_file_not_found(self) -> None:
)

with pytest.raises(NotFoundException) as exception_info:
await action(123, 456, "supersecrettoken")
await action(123, 456, "supersecrettoken", AttachmentTypes.ORIGINAL)

assert str(exception_info.value) == "File not found"

Expand Down

0 comments on commit 81ca5b5

Please sign in to comment.