taiko-web/app.py

177 lines
4.9 KiB
Python
Raw Normal View History

2018-08-27 00:14:56 +08:00
#!/usr/bin/env python2
2018-10-18 05:16:53 +08:00
from __future__ import division
2018-10-11 04:57:29 +08:00
import json
2018-08-27 00:14:56 +08:00
import sqlite3
2018-08-27 20:28:30 +08:00
import re
import os
2018-10-18 05:16:53 +08:00
from flask import Flask, g, jsonify, render_template, request, abort, redirect
2019-02-14 18:31:23 +08:00
from flask_caching import Cache
2018-10-18 05:16:53 +08:00
from ffmpy import FFmpeg
2018-08-27 00:14:56 +08:00
app = Flask(__name__)
2019-02-14 18:31:23 +08:00
try:
app.cache = Cache(app, config={'CACHE_TYPE': 'redis'})
except RuntimeError:
2019-02-14 18:36:39 +08:00
import tempfile
2019-02-14 18:31:23 +08:00
app.cache = Cache(app, config={'CACHE_TYPE': 'filesystem', 'CACHE_DIR': tempfile.gettempdir()})
2018-08-27 00:14:56 +08:00
DATABASE = 'taiko.db'
2019-02-07 02:49:25 +08:00
DEFAULT_URL = 'https://github.com/bui/taiko-web/'
2018-08-27 00:14:56 +08:00
def get_db():
db = getattr(g, '_database', None)
if db is None:
db = g._database = sqlite3.connect(DATABASE)
return db
def query_db(query, args=(), one=False):
cur = get_db().execute(query, args)
rv = cur.fetchall()
cur.close()
return (rv[0] if rv else None) if one else rv
2018-10-28 17:59:49 +08:00
def get_config():
if os.path.isfile('config.json'):
try:
config = json.load(open('config.json', 'r'))
except ValueError:
2019-02-21 06:02:44 +08:00
print('WARNING: Invalid config.json, using default values')
2018-10-28 17:59:49 +08:00
config = {}
else:
2019-02-21 06:02:44 +08:00
print('WARNING: No config.json found, using default values')
2018-10-28 17:59:49 +08:00
config = {}
if not config.get('songs_baseurl'):
config['songs_baseurl'] = ''.join([request.host_url, 'songs']) + '/'
if not config.get('assets_baseurl'):
config['assets_baseurl'] = ''.join([request.host_url, 'assets']) + '/'
2018-12-06 05:55:28 +08:00
config['_version'] = get_version()
2018-10-28 17:59:49 +08:00
return config
2018-12-06 05:47:35 +08:00
def get_version():
2019-02-07 02:49:25 +08:00
version = {'commit': None, 'commit_short': '', 'version': None, 'url': DEFAULT_URL}
2018-12-06 05:47:35 +08:00
if os.path.isfile('version.json'):
2019-02-07 02:49:25 +08:00
try:
ver = json.load(open('version.json', 'r'))
except ValueError:
print('Invalid version.json file')
return version
for key in version.keys():
if ver.get(key):
version[key] = ver.get(key)
2018-12-06 05:47:35 +08:00
return version
2018-08-27 00:14:56 +08:00
@app.teardown_appcontext
def close_connection(exception):
db = getattr(g, '_database', None)
if db is not None:
db.close()
2018-10-11 04:57:29 +08:00
@app.route('/')
2019-02-14 18:36:39 +08:00
@app.cache.cached(timeout=15)
2018-10-11 04:57:29 +08:00
def route_index():
2018-12-06 05:47:35 +08:00
version = get_version()
2018-10-28 17:59:49 +08:00
return render_template('index.html', version=version, config=get_config())
2018-10-11 04:57:29 +08:00
2018-10-18 05:16:53 +08:00
@app.route('/api/preview')
2019-02-14 18:31:23 +08:00
@app.cache.cached(timeout=15)
2018-10-18 05:16:53 +08:00
def route_api_preview():
song_id = request.args.get('id', None)
if not song_id or not re.match('^[0-9]+$', song_id):
abort(400)
song_row = query_db('select * from songs where id = ? and enabled = 1', (song_id,))
if not song_row:
abort(400)
2018-11-19 03:53:37 +08:00
song_type = song_row[0][12]
2019-02-21 06:02:44 +08:00
prev_path = make_preview(song_id, song_type, song_row[0][15])
2018-10-18 05:16:53 +08:00
if not prev_path:
return redirect(get_config()['songs_baseurl'] + '%s/main.mp3' % song_id)
2018-10-18 05:16:53 +08:00
return redirect(get_config()['songs_baseurl'] + '%s/preview.mp3' % song_id)
2018-10-18 05:16:53 +08:00
2018-08-27 00:14:56 +08:00
@app.route('/api/songs')
2019-02-14 18:31:23 +08:00
@app.cache.cached(timeout=15)
2018-08-27 00:14:56 +08:00
def route_api_songs():
2018-08-27 20:28:30 +08:00
songs = query_db('select * from songs where enabled = 1')
2018-11-26 06:42:24 +08:00
2018-09-27 02:30:57 +08:00
raw_categories = query_db('select * from categories')
categories = {}
for cat in raw_categories:
2019-01-25 09:42:05 +08:00
categories[cat[0]] = cat[1]
2018-11-26 06:42:24 +08:00
raw_song_skins = query_db('select * from song_skins')
song_skins = {}
for skin in raw_song_skins:
2018-12-02 23:25:42 +08:00
song_skins[skin[0]] = {'name': skin[1], 'song': skin[2], 'stage': skin[3], 'don': skin[4]}
2018-11-26 06:42:24 +08:00
2018-08-27 20:28:30 +08:00
songs_out = []
for song in songs:
song_id = song[0]
2018-11-11 03:12:29 +08:00
song_type = song[12]
2019-02-21 06:02:44 +08:00
preview = song[15]
2018-11-26 06:42:24 +08:00
category_out = categories[song[11]] if song[11] in categories else ""
2018-12-02 23:25:42 +08:00
song_skin_out = song_skins[song[14]] if song[14] in song_skins else None
2018-09-27 02:30:57 +08:00
songs_out.append({
'id': song_id,
2018-09-27 02:30:57 +08:00
'title': song[1],
2019-01-25 09:42:05 +08:00
'title_lang': song[2],
2018-11-11 03:12:29 +08:00
'subtitle': song[3],
2019-01-25 09:42:05 +08:00
'subtitle_lang': song[4],
2018-09-27 02:30:57 +08:00
'stars': [
2018-11-11 03:12:29 +08:00
song[5], song[6], song[7], song[8], song[9]
2018-09-27 02:30:57 +08:00
],
'preview': preview,
2019-01-25 09:42:05 +08:00
'category': category_out,
'type': song_type,
2018-11-26 06:42:24 +08:00
'offset': song[13],
'song_skin': song_skin_out,
'volume': song[16]
2018-09-27 02:30:57 +08:00
})
2018-08-27 00:14:56 +08:00
2018-08-27 20:28:30 +08:00
return jsonify(songs_out)
2018-08-27 00:14:56 +08:00
2018-10-28 05:42:28 +08:00
@app.route('/api/config')
2019-02-14 18:31:23 +08:00
@app.cache.cached(timeout=15)
2018-10-28 05:42:28 +08:00
def route_api_config():
2018-10-28 17:59:49 +08:00
config = get_config()
2018-10-28 05:42:28 +08:00
return jsonify(config)
2019-02-21 06:02:44 +08:00
def make_preview(song_id, song_type, preview):
2018-10-18 05:16:53 +08:00
song_path = 'public/songs/%s/main.mp3' % song_id
prev_path = 'public/songs/%s/preview.mp3' % song_id
if os.path.isfile(song_path) and not os.path.isfile(prev_path):
2019-02-21 06:42:18 +08:00
if not preview or preview <= 0:
2019-02-07 02:49:25 +08:00
print('Skipping #%s due to no preview' % song_id)
2018-10-18 05:16:53 +08:00
return False
2019-02-07 02:49:25 +08:00
print('Making preview.mp3 for song #%s' % song_id)
2018-10-18 05:16:53 +08:00
ff = FFmpeg(inputs={song_path: '-ss %s' % preview},
outputs={prev_path: '-codec:a libmp3lame -ar 32000 -b:a 92k -y -loglevel panic'})
2018-10-18 05:16:53 +08:00
ff.run()
return prev_path
2018-08-27 00:14:56 +08:00
if __name__ == '__main__':
2018-08-27 20:28:30 +08:00
app.run(port=34801)