mirror of
https://github.com/spl0k/supysonic.git
synced 2024-12-22 08:56:17 +00:00
Add cover art to downloaded zip files
- Only includes cover art in the zip file if it's provided separately from the files (ie. doesn't extract it from the tracks) - Refactors the existing code for implementing the `/getCoverArt` API endpoint to allow it to be reused.
This commit is contained in:
parent
e8d3f164b0
commit
5490189484
@ -27,6 +27,7 @@ from zipstream import ZipFile
|
||||
|
||||
from ..cache import CacheMiss
|
||||
from ..db import Track, Album, Folder, now
|
||||
from ..covers import EXTENSIONS
|
||||
|
||||
from . import get_entity, get_entity_id, api_routing
|
||||
from .exceptions import (
|
||||
@ -249,14 +250,63 @@ def download_media():
|
||||
except ObjectNotFound:
|
||||
raise NotFound("Folder")
|
||||
|
||||
# Stream a zip of the tracks + cover art to the client
|
||||
z = ZipFile(compression=ZIP_DEFLATED)
|
||||
for track in rv.tracks:
|
||||
z.write(track.path, os.path.basename(track.path))
|
||||
|
||||
cover_path = _cover_from_collection(rv, extract=False)
|
||||
if cover_path:
|
||||
z.write(cover_path, os.path.basename(cover_path))
|
||||
|
||||
resp = Response(z, mimetype="application/zip")
|
||||
resp.headers["Content-Disposition"] = "attachment; filename={}.zip".format(rv.name)
|
||||
return resp
|
||||
|
||||
|
||||
def _cover_from_track(tid):
|
||||
"""Extract and return a path to a track's cover art
|
||||
|
||||
Returns None if no cover art is available.
|
||||
"""
|
||||
cache = current_app.cache
|
||||
cache_key = "{}-cover".format(tid)
|
||||
try:
|
||||
return cache.get(cache_key)
|
||||
except CacheMiss:
|
||||
obj = Track[tid]
|
||||
try:
|
||||
return cache.set(cache_key, mediafile.MediaFile(obj.path).art)
|
||||
except mediafile.UnreadableFileError:
|
||||
return None
|
||||
|
||||
|
||||
def _cover_from_collection(obj, extract=True):
|
||||
"""Get a path to cover art from a collection (Album, Folder)
|
||||
|
||||
If `extract` is True, will fall back to extracting cover art from tracks
|
||||
Returns None if no cover art is available.
|
||||
"""
|
||||
cover_path = None
|
||||
|
||||
if isinstance(obj, Folder) and obj.cover_art:
|
||||
cover_path = os.path.join(obj.path, obj.cover_art)
|
||||
|
||||
elif isinstance(obj, Album):
|
||||
track_with_folder_cover = obj.tracks.select(lambda t: t.folder.cover_art is not None).first()
|
||||
if track_with_folder_cover is not None:
|
||||
cover_path = _cover_from_collection(track_with_folder_cover.folder)
|
||||
|
||||
if not cover_path and extract:
|
||||
track_with_embedded = obj.tracks.select(lambda t: t.has_art).first()
|
||||
if track_with_embedded is not None:
|
||||
cover_path = _cover_from_track(track_with_embedded.id)
|
||||
|
||||
if not cover_path or not os.path.isfile(cover_path):
|
||||
return None
|
||||
return cover_path
|
||||
|
||||
|
||||
@api_routing("/getCoverArt")
|
||||
def cover_art():
|
||||
cache = current_app.cache
|
||||
@ -267,39 +317,38 @@ def cover_art():
|
||||
except GenericError:
|
||||
fid = None
|
||||
try:
|
||||
tid = get_entity_id(Track, eid)
|
||||
uid = get_entity_id(Track, eid)
|
||||
except GenericError:
|
||||
tid = None
|
||||
uid = None
|
||||
|
||||
if not fid and not tid:
|
||||
if not fid and not uid:
|
||||
raise GenericError("Invalid ID")
|
||||
|
||||
cover_path = None
|
||||
if fid and Folder.exists(id=eid):
|
||||
res = get_entity(Folder)
|
||||
if not res.cover_art or not os.path.isfile(
|
||||
os.path.join(res.path, res.cover_art)
|
||||
):
|
||||
raise NotFound("Cover art")
|
||||
cover_path = os.path.join(res.path, res.cover_art)
|
||||
elif tid and Track.exists(id=eid):
|
||||
cache_key = "{}-cover".format(eid)
|
||||
try:
|
||||
cover_path = cache.get(cache_key)
|
||||
except CacheMiss:
|
||||
res = get_entity(Track)
|
||||
try:
|
||||
art = mediafile.MediaFile(res.path).art
|
||||
except mediafile.UnreadableFileError:
|
||||
raise NotFound("Cover art")
|
||||
cover_path = cache.set(cache_key, art)
|
||||
cover_path = _cover_from_collection(get_entity(Folder))
|
||||
elif uid and Track.exists(id=eid):
|
||||
cover_path = _cover_from_track(eid)
|
||||
elif uid and Album.exists(id=uid):
|
||||
cover_path = _cover_from_collection(get_entity(Album))
|
||||
else:
|
||||
raise NotFound("Entity")
|
||||
|
||||
if not cover_path:
|
||||
raise NotFound("Cover art")
|
||||
|
||||
size = request.values.get("size")
|
||||
if size:
|
||||
size = int(size)
|
||||
else:
|
||||
return send_file(cover_path)
|
||||
# If the cover was extracted from a track it won't have an accurate
|
||||
# extension for Flask to derive the mimetype from - derive it from the
|
||||
# contents instead.
|
||||
mimetype = None
|
||||
if uid and os.path.splitext(cover_path)[1].lower() not in EXTENSIONS:
|
||||
with Image.open(cover_path) as im:
|
||||
mimetype = "image/{}".format(im.format.lower())
|
||||
return send_file(cover_path, mimetype=mimetype)
|
||||
|
||||
with Image.open(cover_path) as im:
|
||||
mimetype = "image/{}".format(im.format.lower())
|
||||
|
Loading…
Reference in New Issue
Block a user