|
- # Created 2019-09-10 by NGnius
- from flask import Flask, Blueprint, jsonify, request, send_file
- import sqlite3 as dblib
- from hashlib import sha512
- from os import getcwd
- from threading import get_ident
- from time import time
- from datetime import datetime
-
- database_path = 'rxsm-update.db'
-
- database_connections = dict()
- app = Flask('rxsm-update-server')
-
- def get_or_create_connection():
- global database_connections, database_path
- thread_id = get_ident()
- if thread_id not in database_connections:
- database_connections[thread_id] = dblib.connect(database_path)
- return database_connections[thread_id]
-
- @app.route('/release', methods=['POST'])
- def create_or_modify_release():
- try:
- req_json = request.get_json(force=True)
- token = req_json['token']
- version = req_json['version']
- platform = req_json['platform']
- url = req_json['url']
- except:
- return jsonify({'status':400, 'reason':'Invalid request; missing parameter or invalid JSON'}), 400
-
- with open('.token', 'r') as f:
- master_token = f.read()
- if sha512(token.encode()).hexdigest() != master_token:
- return jsonify({'status':403, 'reason':'Permission denied; invalid token'}), 403
-
- db_connection = get_or_create_connection()
- cursor = db_connection.cursor()
- cursor.execute('SELECT version FROM releases WHERE version=? AND platform=?', (version, platform))
- result = cursor.fetchone()
- operation = ''
- if result is None:
- cursor.execute('INSERT INTO releases (version, platform, url, created_date) VALUES (?,?,?,?)', (version, platform, url, int(time())))
- operation = 'create'
- else:
- cursor.execute('UPDATE releases SET url=?, created_date=? WHERE version=? and platform=?', (url, int(time()), version, platform))
- operation = 'update'
- db_connection.commit()
-
- return jsonify({'status':200, 'reason':'Version %s %sd successfully' % (version, operation)}), 200
-
- @app.route('/db', methods=['GET'])
- @app.route('/database', methods=['GET'])
- def download_database():
- return send_file(database_path, cache_timeout=1, as_attachment=True, attachment_filename='rxsm.sqlite')
-
- @app.route('/update', methods=['POST'])
- @app.route('/check-for-update', methods=['POST'])
- def check_for_update():
- start = time()
- db_connection = get_or_create_connection()
- cursor = db_connection.cursor()
- collect_anonymous_stats = str(request.headers.get('DNT')).strip() != '1'
- # hit counter
- if collect_anonymous_stats:
- today = datetime.today().date().isoformat()
- cursor.execute('SELECT count FROM hits WHERE date=?', (today,))
- result = cursor.fetchone()
- if result is not None:
- cursor.execute('UPDATE hits SET count=count+1 WHERE date=?', (today,))
- else:
- cursor.execute('INSERT INTO hits (date, count) VALUES (?,?)', (today, 1))
- db_connection.commit()
- # update lookup
- try:
- req_json = request.get_json(force=True)
- current_version = req_json['version']
- platform = req_json['platform']
- except:
- return jsonify({'status':400, 'reason':'Invalid request; missing parameter or invalid JSON'}), 400
- cursor.execute('SELECT version, url FROM releases WHERE platform=? ORDER BY created_date DESC', (platform,))
- result = cursor.fetchone()
- if result is None:
- return jsonify({'status':404, 'reason':'Platform not found (%ss)' % (time()-start), 'url':'', 'out-of-date':False}), 404
- elif result[0] != current_version:
- return jsonify({'status':200, 'reason':'Ok (%ss)' % (time()-start), 'url':result[1], 'out-of-date':True}), 200
- else:
- return jsonify({'status':200, 'reason':'Ok (%ss)' % (time()-start), 'url':result[1], 'out-of-date':False}), 200
-
- # always try to create db table(s)
- conn = get_or_create_connection()
- cursor = conn.cursor()
- cursor.execute(\
- '''CREATE TABLE IF NOT EXISTS releases (
- id INTEGER PRIMARY KEY,
- version TEXT UNIQUE NOT NULL,
- url TEXT NOT NULL,
- platform TEXT NOT NULL,
- created_date INTEGER NOT NULL
- )''' )
- cursor.execute(\
- '''CREATE TABLE IF NOT EXISTS hits (
- date TEXT UNIQUE NOT NULL,
- count INTEGER NOT NULL
- )''' )
- conn.commit()
-
- if __name__ == '__main__':
- print('Working directory: ' + getcwd())
- app.run(host='0.0.0.0', port=9080, threaded=True)
|