1
0
mirror of https://github.com/spl0k/supysonic.git synced 2024-11-14 22:22:18 +00:00
supysonic/api/media.py

340 lines
11 KiB
Python
Raw Normal View History

2012-10-20 18:05:39 +00:00
# coding: utf-8
2014-03-02 17:31:32 +00:00
# This file is part of Supysonic.
#
# Supysonic is a Python implementation of the Subsonic server API.
# Copyright (C) 2013 Alban 'spl0k' Féron
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
2013-10-15 11:14:20 +00:00
from flask import request, send_file, Response
2013-12-27 21:53:07 +00:00
import requests
2012-12-02 15:42:25 +00:00
import os.path
2013-06-07 13:31:30 +00:00
from PIL import Image
2013-10-15 11:14:20 +00:00
import subprocess
2013-12-09 18:06:14 +00:00
import codecs
2013-12-27 21:53:07 +00:00
from xml.etree import ElementTree
import shlex
import fnmatch
import mimetypes
from mediafile import MediaFile
import mutagen
2012-11-23 16:13:25 +00:00
2013-10-15 11:14:20 +00:00
import config, scanner
2012-11-23 16:13:25 +00:00
from web import app
from db import Track, Album, Artist, Folder, User, ClientPrefs, now, session
2014-03-04 21:56:53 +00:00
from . import get_entity
2012-10-20 18:05:39 +00:00
from sqlalchemy import func
2012-10-20 18:05:39 +00:00
from flask import g
def after_this_request(func):
if not hasattr(g, 'call_after_request'):
g.call_after_request = []
g.call_after_request.append(func)
return func
@app.after_request
def per_request_callbacks(response):
for func in getattr(g, 'call_after_request', ()):
response = func(response)
return response
2013-10-15 11:14:20 +00:00
def prepare_transcoding_cmdline(base_cmdline, input_file, input_format, output_format, output_bitrate):
if not base_cmdline:
return None
return base_cmdline.replace('%srcpath', '"'+input_file+'"').replace('%srcfmt', input_format).replace('%outfmt', output_format).replace('%outrate', str(output_bitrate))
2013-10-15 11:14:20 +00:00
2012-11-22 13:51:43 +00:00
@app.route('/rest/stream.view', methods = [ 'GET', 'POST' ])
2012-10-20 18:05:39 +00:00
def stream_media():
@after_this_request
def add_header(response):
if 'X-Sendfile' in response.headers:
xsendfile = response.headers['X-Sendfile'] or ''
redirect = config.get('base', 'accel-redirect')
if redirect and xsendfile:
response.headers['X-Accel-Charset'] = 'utf-8'
response.headers['X-Accel-Redirect'] = redirect + xsendfile.encode('UTF8')
app.logger.debug('X-Accel-Redirect: ' + redirect + xsendfile)
return response
def transcode(process):
try:
for chunk in iter(process.stdout.readline, ''):
yield chunk
process.wait()
except:
app.logger.debug('transcoding timeout, killing process')
process.terminate()
process.wait()
2012-12-02 15:42:25 +00:00
status, res = get_entity(request, Track)
2012-12-02 15:42:25 +00:00
if not status:
return res
2012-10-20 18:05:39 +00:00
maxBitRate, format, timeOffset, size, estimateContentLength, client = map(request.args.get, [ 'maxBitRate', 'format', 'timeOffset', 'size', 'estimateContentLength', 'c' ])
2013-10-15 08:32:35 +00:00
if format:
format = format.lower()
2012-10-20 18:05:39 +00:00
2013-10-15 11:14:20 +00:00
do_transcoding = False
src_suffix = res.suffix()
dst_suffix = src_suffix
2013-10-15 11:14:20 +00:00
dst_bitrate = res.bitrate
dst_mimetype = mimetypes.guess_type('a.' + src_suffix)
2013-10-15 11:14:20 +00:00
if maxBitRate:
try:
maxBitRate = int(maxBitRate)
except:
return request.error_formatter(0, 'Invalid bitrate value')
2012-10-20 18:05:39 +00:00
if dst_bitrate > maxBitRate and maxBitRate != 0:
2013-10-15 11:14:20 +00:00
do_transcoding = True
dst_bitrate = maxBitRate
if format and format != 'raw' and format != src_suffix:
do_transcoding = True
dst_suffix = format
dst_mimetype = mimetypes.guess_type(dst_suffix)
2013-10-15 11:14:20 +00:00
if client:
prefs = ClientPrefs.query.get((request.user.id, client))
if not prefs:
prefs = ClientPrefs(user_id = request.user.id, client_name = client)
session.add(prefs)
2012-10-20 18:05:39 +00:00
if prefs.format:
dst_suffix = prefs.format
if prefs.bitrate and prefs.bitrate < dst_bitrate:
dst_bitrate = prefs.bitrate
2013-10-15 11:14:20 +00:00
if not format and src_suffix == 'flac':
dst_suffix = 'ogg'
dst_bitrate = 320
dst_mimetype = 'audio/ogg'
do_transcoding = True
duration = mutagen.File(res.path).info.length
app.logger.debug('Serving file: ' + res.path + '\n\tDuration of file: ' + str(duration))
2013-10-15 11:14:20 +00:00
if do_transcoding:
transcoder = config.get('transcoding', 'transcoder_{}_{}'.format(src_suffix, dst_suffix))
2013-10-15 11:14:20 +00:00
decoder = config.get('transcoding', 'decoder_' + src_suffix) or config.get('transcoding', 'decoder')
encoder = config.get('transcoding', 'encoder_' + dst_suffix) or config.get('transcoding', 'encoder')
2013-10-15 11:14:20 +00:00
if not transcoder and (not decoder or not encoder):
transcoder = config.get('transcoding', 'transcoder')
if not transcoder:
return request.error_formatter(0, 'No way to transcode from {} to {}'.format(src_suffix, dst_suffix))
2013-10-15 11:14:20 +00:00
transcoder, decoder, encoder = map(lambda x: prepare_transcoding_cmdline(x, res.path, src_suffix, dst_suffix, dst_bitrate), [ transcoder, decoder, encoder ])
decoder = map(lambda s: s.decode('UTF8'), shlex.split(decoder.encode('utf8')))
encoder = map(lambda s: s.decode('UTF8'), shlex.split(encoder.encode('utf8')))
transcoder = map(lambda s: s.decode('UTF8'), shlex.split(transcoder.encode('utf8')))
app.logger.debug(str( decoder ) + '\n' + str( encoder ) + '\n' + str(transcoder))
if '|' in transcoder:
pipe_index = transcoder.index('|')
decoder = transcoder[:pipe_index]
encoder = transcoder[pipe_index+1:]
transcoder = None
app.logger.debug('decoder' + str(decoder) + '\nencoder' + str(encoder))
2013-10-20 15:27:20 +00:00
try:
if transcoder:
2013-11-05 16:47:08 +00:00
app.logger.warn('transcoder: '+str(transcoder))
proc = subprocess.Popen(transcoder, stdout = subprocess.PIPE, shell=False)
2013-10-20 15:27:20 +00:00
else:
dec_proc = subprocess.Popen(decoder, stdout = subprocess.PIPE, shell=False)
proc = subprocess.Popen(encoder, stdin = dec_proc.stdout, stdout = subprocess.PIPE, shell=False)
response = Response(transcode(proc), 200, {'Content-Type': dst_mimetype, 'X-Content-Duration': str(duration)})
2013-10-20 15:27:20 +00:00
except:
return request.error_formatter(0, 'Error while running the transcoding process')
2013-10-15 11:14:20 +00:00
else:
app.logger.warn('no transcode')
response = send_file(res.path)
response.headers['Content-Type'] = dst_mimetype
response.headers['Accept-Ranges'] = 'bytes'
response.headers['X-Content-Duration'] = str(duration)
2013-10-15 11:14:20 +00:00
res.play_count = res.play_count + 1
res.last_play = now()
request.user.last_play = res
request.user.last_play_date = now()
session.commit()
2013-10-15 11:14:20 +00:00
return response
2012-10-20 18:05:39 +00:00
2013-06-12 19:29:42 +00:00
@app.route('/rest/download.view', methods = [ 'GET', 'POST' ])
def download_media():
status, res = get_entity(request, Track)
if not status:
return res
return send_file(res.path)
2012-11-22 13:51:43 +00:00
@app.route('/rest/getCoverArt.view', methods = [ 'GET', 'POST' ])
2012-11-11 20:39:26 +00:00
def cover_art():
# Speed up the file transfer
@after_this_request
def add_header(response):
response.headers['Content-Type'] = 'image/jpeg'
if 'X-Sendfile' in response.headers:
redirect = response.headers['X-Sendfile'] or ''
xsendfile = config.get('base', 'accel-redirect')
if redirect and xsendfile:
response.headers['X-Accel-Redirect'] = xsendfile + redirect
app.logger.debug('X-Accel-Redirect: ' + xsendfile + redirect)
return response
# retrieve folder from database
2012-12-02 15:42:25 +00:00
status, res = get_entity(request, Folder)
2012-12-02 15:42:25 +00:00
if not status:
return res
# Check the folder id given for jpgs
app.logger.debug('Cover Art Check: ' + res.path + '/*.jp*g')
coverfile = os.listdir(res.path)
coverfile = fnmatch.filter(coverfile, '*.jp*g')
app.logger.debug('Found Images: ' + str(coverfile))
# when there is not a jpeg in the folder check files for embedded art
if not coverfile:
app.logger.debug('No Art Found in Folder, Checking Files!')
for tr in res.tracks:
app.logger.debug('Checking ' + tr.path + ' For Artwork')
try:
mf = MediaFile(tr.path)
coverfile = getattr(mf, 'art')
if coverfile is not None:
return coverfile
except:
app.logger.debug('Problem reading embedded art')
tr.folder.has_cover_art = False
session.commit()
2014-02-01 07:34:18 +00:00
return request.error_formatter(70, 'Cover art not found'), 404
2012-11-11 20:39:26 +00:00
# pick the first image found
# TODO: prefer cover
coverfile = coverfile[0]
2012-11-11 20:39:26 +00:00
size = request.args.get('size')
if size:
try:
size = int(size)
except:
return request.error_formatter(0, 'Invalid size value'), 500
2012-11-11 20:39:26 +00:00
else:
app.logger.debug('Serving cover art: ' + res.path + coverfile)
return send_file(os.path.join(res.path, coverfile))
2012-11-11 20:39:26 +00:00
im = Image.open(os.path.join(res.path, coverfile))
2012-11-11 20:39:26 +00:00
if size > im.size[0] and size > im.size[1]:
app.logger.debug('Serving cover art: ' + res.path + coverfile)
return send_file(os.path.join(res.path, coverfile))
2012-11-11 20:39:26 +00:00
size_path = os.path.join(config.get('base', 'cache_dir'), str(size))
2012-12-02 15:42:25 +00:00
path = os.path.join(size_path, str(res.id))
2012-11-11 20:39:26 +00:00
if os.path.exists(path):
app.logger.debug('Serving cover art: ' + path)
2012-11-11 20:39:26 +00:00
return send_file(path)
if not os.path.exists(size_path):
os.makedirs(size_path)
im.thumbnail([size, size], Image.ANTIALIAS)
im.save(path, 'JPEG')
app.logger.debug('Serving cover art: ' + path)
2012-11-11 20:39:26 +00:00
return send_file(path)
@app.route('/rest/getLyrics.view', methods = [ 'GET', 'POST' ])
def lyrics():
artist, title = map(request.args.get, [ 'artist', 'title' ])
if not artist:
return request.error_formatter(10, 'Missing artist parameter')
if not title:
return request.error_formatter(10, 'Missing title parameter')
query = Track.query.join(Album, Artist).filter(func.lower(Track.title) == title.lower() and func.lower(Artist.name) == artist.lower())
for track in query:
lyrics_path = os.path.splitext(track.path)[0] + '.txt'
if os.path.exists(lyrics_path):
2013-12-09 18:06:14 +00:00
app.logger.debug('Found lyrics file: ' + lyrics_path)
try:
lyrics = read_file_as_unicode(lyrics_path)
except UnicodeError:
# Lyrics file couldn't be decoded. Rather than displaying an error, try with the potential next files or
# return no lyrics. Log it anyway.
app.logger.warn('Unsupported encoding for lyrics file ' + lyrics_path)
continue
return request.formatter({ 'lyrics': {
'artist': track.album.artist.name,
'title': track.title,
'_value_': lyrics
} })
2013-12-27 21:53:07 +00:00
try:
r = requests.get("http://api.chartlyrics.com/apiv1.asmx/SearchLyricDirect",
params = { 'artist': artist, 'song': title })
root = ElementTree.fromstring(r.content)
ns = { 'cl': 'http://api.chartlyrics.com/' }
return request.formatter({ 'lyrics': {
'artist': root.find('cl:LyricArtist', namespaces = ns).text,
'title': root.find('cl:LyricSong', namespaces = ns).text,
'_value_': root.find('cl:Lyric', namespaces = ns).text
} })
except requests.exceptions.RequestException, e:
app.logger.warn('Error while requesting the ChartLyrics API: ' + str(e))
return request.formatter({ 'lyrics': {} })
2013-12-09 18:06:14 +00:00
def read_file_as_unicode(path):
""" Opens a file trying with different encodings and returns the contents as a unicode string """
encodings = [ 'utf-8', 'latin1' ] # Should be extended to support more encodings
for enc in encodings:
try:
contents = codecs.open(path, 'r', encoding = enc).read()
app.logger.debug('Read file {} with {} encoding'.format(path, enc))
# Maybe save the encoding somewhere to prevent going through this loop each time for the same file
return contents
except UnicodeError:
pass
# Fallback to ASCII
app.logger.debug('Reading file {} with ascii encoding'.format(path))
return unicode(open(path, 'r').read())