Compare commits

...

30 Commits

Author SHA1 Message Date
2a8040642a Merge branch 'legacy' into feature/legacy/mariadb_database 2023-03-28 01:26:24 +02:00
cbf7aad8ac Fix Type errors 2023-03-28 01:25:33 +02:00
d5a21b82de Add GitHub Link to footer 2023-03-28 01:00:41 +02:00
551536bcb4 Update bootstrap-tables 2023-03-28 01:00:21 +02:00
b8220732ee Introduce "dev environment" and dotenv
There are now two dockerfiles. One for production, one for development.
All configuration is now also handled through dotenv files,
which these dotenv files, as well as the included VSCode launch tasks
use.
2023-03-28 00:54:04 +02:00
831166f38b Intall dependencies using requirements.txt 2023-03-27 23:38:22 +02:00
41a24ad9ce readd main.py 2023-03-27 01:02:26 +02:00
6efb234a1c Merge pull request #39 from PhoenixTwoFive/feature/legacy/mariadb_database
Feature/legacy/mariadb database
2022-10-08 19:51:56 +02:00
698c3717fd Remove outdated workflows 2022-10-08 19:50:35 +02:00
8dd90b728c MySQL Working, bug fixes. 2022-10-08 19:47:29 +02:00
a22960eae9 Fix DB Connection 2022-07-15 15:54:15 +02:00
bc44985fed Fix DBCONNSTRING 2022-07-15 15:35:58 +02:00
72914109c0 prune requirements 2022-07-15 15:33:42 +02:00
57ced69ddd Load DB Connection from ENV 2022-07-15 15:27:36 +02:00
fc78bdc4fe Fix types 2022-07-15 15:09:48 +02:00
adfd120636 Update package names for 22.04 2022-07-15 15:01:07 +02:00
67a9552fee Update ubuntu to get newer mariadb connector 2022-07-15 14:55:14 +02:00
3ae2803fcc Remove mysql and install mariadb 2022-07-15 14:51:05 +02:00
2138189dff python3 is still not default on Ubuntu? 2022-07-15 14:44:26 +02:00
9907a19066 Test 2022-07-15 14:43:16 +02:00
2a18f3dc2c Completely remove mariadb libs to hopefully fix conflicts 2022-07-15 14:36:47 +02:00
69e252e28a Add missing libraries to install 2022-07-15 14:35:52 +02:00
8151e615c0 Test 2022-07-15 14:34:07 +02:00
2a92c39450 Test 2022-07-15 14:32:51 +02:00
b88aac69e6 Fix.. 2022-07-15 14:32:06 +02:00
c9ad755a04 Fix? 2022-07-15 14:29:18 +02:00
3527ba4fd9 Fix dependencies 2022-07-15 14:27:39 +02:00
fb9677dc88 Fix github workflow 2022-07-15 14:23:17 +02:00
97dd80b03a Add or update the Azure App Service build and deployment workflow config 2022-07-15 14:16:06 +02:00
faad60346f Intial changes for MariaDB 2022-07-15 14:01:02 +02:00
15 changed files with 241 additions and 169 deletions

14
.env.dev Normal file
View File

@ -0,0 +1,14 @@
# MariaDB
MARIADB_ROOT_PASSWORD=mariadb_root_password
MARIADB_ROOT_HOST=localhost
MARIADB_DATABASE=karaoqueue
MARIADB_USER=karaoqueue
MARIADB_PASSWORD=mariadb_karaoqueue_password
# Karaoqueue
DEPLOYMENT_PLATFORM=Docker
DBSTRING=mysql://karaoqueue:mariadb_karaoqueue_password@127.0.0.1:3306/karaoqueue
BASIC_AUTH_USERNAME=admin
BASIC_AUTH_PASSWORD=change_me
ENTRY_QUOTA=3
MAX_QUEUE=20

5
.gitignore vendored
View File

@ -137,4 +137,7 @@ data/
node_modules/ node_modules/
# Version identification file # Version identification file
.version .version
# Docker secrets
secrets.yml

14
.vscode/launch.json vendored
View File

@ -5,7 +5,7 @@
"version": "0.2.0", "version": "0.2.0",
"configurations": [ "configurations": [
{ {
"preLaunchTask": "versiondump", "preLaunchTask": "mariadb",
"name": "Python: Flask", "name": "Python: Flask",
"type": "python", "type": "python",
"cwd": "${workspaceFolder}/backend", "cwd": "${workspaceFolder}/backend",
@ -14,8 +14,9 @@
"env": { "env": {
"FLASK_APP": "backend/app.py", "FLASK_APP": "backend/app.py",
"FLASK_ENV": "development", "FLASK_ENV": "development",
"FLASK_DEBUG": "1" "FLASK_DEBUG": "1",
}, },
"envFile": "${workspaceFolder}/.env.dev",
"args": [ "args": [
"run", "run",
"--no-debugger", "--no-debugger",
@ -24,7 +25,7 @@
"jinja": true "jinja": true
}, },
{ {
"preLaunchTask": "versiondump", "preLaunchTask": "mariadb",
"name": "Python: Flask (with reload)", "name": "Python: Flask (with reload)",
"type": "python", "type": "python",
"cwd": "${workspaceFolder}/backend", "cwd": "${workspaceFolder}/backend",
@ -35,6 +36,7 @@
"FLASK_ENV": "development", "FLASK_ENV": "development",
"FLASK_DEBUG": "1" "FLASK_DEBUG": "1"
}, },
"envFile": "${workspaceFolder}/.env.dev",
"args": [ "args": [
"run", "run",
"--no-debugger" "--no-debugger"
@ -42,7 +44,7 @@
"jinja": true "jinja": true
}, },
{ {
"preLaunchTask": "versiondump", "preLaunchTask": "mariadb",
"name": "Python: Flask (with reload, externally reachable)", "name": "Python: Flask (with reload, externally reachable)",
"type": "python", "type": "python",
"cwd": "${workspaceFolder}/backend", "cwd": "${workspaceFolder}/backend",
@ -53,6 +55,7 @@
"FLASK_ENV": "development", "FLASK_ENV": "development",
"FLASK_DEBUG": "1" "FLASK_DEBUG": "1"
}, },
"envFile": "${workspaceFolder}/.env.dev",
"args": [ "args": [
"run", "run",
"--no-debugger", "--no-debugger",
@ -61,7 +64,7 @@
"jinja": true "jinja": true
}, },
{ {
"preLaunchTask": "versiondump", "preLaunchTask": "mariadb",
"name": "Python: Flask (externally reachable)", "name": "Python: Flask (externally reachable)",
"type": "python", "type": "python",
"cwd": "${workspaceFolder}/backend", "cwd": "${workspaceFolder}/backend",
@ -72,6 +75,7 @@
"FLASK_ENV": "development", "FLASK_ENV": "development",
"FLASK_DEBUG": "1" "FLASK_DEBUG": "1"
}, },
"envFile": "${workspaceFolder}/.env.dev",
"args": [ "args": [
"run", "run",
"--no-debugger", "--no-debugger",

7
.vscode/tasks.json vendored
View File

@ -8,6 +8,13 @@
"type": "shell", "type": "shell",
"command": "echo \"$(git rev-parse --abbrev-ref HEAD)-$(git describe)\"> ${workspaceFolder}/backend/.version", "command": "echo \"$(git rev-parse --abbrev-ref HEAD)-$(git describe)\"> ${workspaceFolder}/backend/.version",
"problemMatcher": [] "problemMatcher": []
},
{
"label": "mariadb",
"type": "shell",
"command": "docker-compose -f docker-compose.yml up --remove-orphans",
"isBackground": true,
"activeOnStart": false
} }
] ]
} }

5
Dockerfile Normal file
View File

@ -0,0 +1,5 @@
FROM tiangolo/uwsgi-nginx-flask:python3.7
COPY ./backend /app
RUN pip install -r /app/requirements.txt

View File

@ -1,11 +0,0 @@
FROM tiangolo/uwsgi-nginx-flask:python3.7
RUN pip install requests
RUN pip install pandas
RUN pip install Flask-BasicAuth
RUN pip install bs4
COPY ./app /app

View File

@ -1,4 +1,5 @@
from flask import Flask, render_template, Response, abort, request, redirect, send_from_directory from flask import Flask, render_template, abort, request, redirect, send_from_directory, jsonify
from flask.wrappers import Request, Response
import helpers import helpers
import database import database
import data_adapters import data_adapters
@ -9,7 +10,7 @@ from helpers import nocache
app = Flask(__name__, static_url_path='/static') app = Flask(__name__, static_url_path='/static')
basic_auth = BasicAuth(app) basic_auth = BasicAuth(app)
accept_entries = False accept_entries = True
@app.route("/") @app.route("/")
def home(): def home():
@ -51,8 +52,8 @@ def enqueue():
abort(400) abort(400)
name = request.json['name'] name = request.json['name']
song_id = request.json['id'] song_id = request.json['id']
if database.check_queue_length() < app.config['MAX_QUEUE']: if database.check_queue_length() < int(app.config['MAX_QUEUE']):
if database.check_entry_quota(client_id) < app.config['ENTRY_QUOTA']: if database.check_entry_quota(client_id) < int(app.config['ENTRY_QUOTA']):
database.add_entry(name, song_id, client_id) database.add_entry(name, song_id, client_id)
return Response('{"status":"OK"}', mimetype='text/json') return Response('{"status":"OK"}', mimetype='text/json')
else: else:
@ -81,12 +82,12 @@ def settings():
def settings_post(): def settings_post():
entryquota = request.form.get("entryquota") entryquota = request.form.get("entryquota")
maxqueue = request.form.get("maxqueue") maxqueue = request.form.get("maxqueue")
if entryquota.isnumeric() and int(entryquota) > 0: if entryquota.isnumeric() and int(entryquota) > 0: # type: ignore
app.config['ENTRY_QUOTA'] = int(entryquota) app.config['ENTRY_QUOTA'] = int(entryquota) # type: ignore
else: else:
abort(400) abort(400)
if maxqueue.isnumeric and int(maxqueue) > 0: if maxqueue.isnumeric and int(maxqueue) > 0: # type: ignore
app.config['MAX_QUEUE'] = int(maxqueue) app.config['MAX_QUEUE'] = int(maxqueue) # type: ignore
else: else:
abort(400) abort(400)
@ -131,8 +132,8 @@ def get_song_completions(input_string=""):
input_string = request.args.get('search', input_string) input_string = request.args.get('search', input_string)
if input_string != "": if input_string != "":
print(input_string) print(input_string)
list = database.get_song_completions(input_string=input_string) result = [list(x) for x in database.get_song_completions(input_string=input_string)]
return Response(json.dumps(list, ensure_ascii=False).encode('utf-8'), mimetype='text/json') return jsonify(result)
else: else:
return 400 return 400
@ -229,6 +230,7 @@ def admin():
@app.before_first_request @app.before_first_request
def activate_job(): def activate_job():
helpers.load_dbconfig(app)
helpers.load_version(app) helpers.load_version(app)
helpers.create_data_directory() helpers.create_data_directory()
database.create_entry_table() database.create_entry_table()

View File

@ -1,190 +1,166 @@
# -*- coding: utf_8 -*- # -*- coding: utf_8 -*-
import sqlite3 from email.mime import base
from MySQLdb import Connection
from sqlalchemy import create_engine, engine
import pandas import pandas
from io import StringIO from io import StringIO
from flask import current_app
song_table = "songs" song_table = "songs"
entry_table = "entries" entry_table = "entries"
index_label = "Id" index_label = "Id"
done_table = "done_songs" done_table = "done_songs"
sql_engine = None
def open_db():
conn = sqlite3.connect("/tmp/karaoqueue.db") def get_db_engine() -> engine.base.Engine:
conn.execute('PRAGMA encoding = "UTF-8";') global sql_engine
return conn if (not sql_engine):
print(current_app.config.get("DBCONNSTRING"))
sql_engine = create_engine(current_app.config.get("DBCONNSTRING")) # type: ignore
return sql_engine
def import_songs(song_csv): def import_songs(song_csv):
print("Start importing Songs...") print("Start importing Songs...")
df = pandas.read_csv(StringIO(song_csv), sep=';') df = pandas.read_csv(StringIO(song_csv), sep=';')
conn = open_db() with get_db_engine().connect() as conn:
cur = conn.cursor() df.to_sql(song_table, conn, if_exists='replace',
df.to_sql(song_table, conn, if_exists='replace', index=False)
index=False) cur = conn.execute("SELECT Count(Id) FROM songs")
cur.execute("SELECT Count(Id) FROM songs") num_songs = cur.fetchone()[0] # type: ignore
num_songs = cur.fetchone()[0]
conn.close()
print("Imported songs ({} in Database)".format(num_songs)) print("Imported songs ({} in Database)".format(num_songs))
return("Imported songs ({} in Database)".format(num_songs)) return("Imported songs ({} in Database)".format(num_songs))
def create_entry_table(): def create_entry_table():
conn = open_db() with get_db_engine().connect() as conn:
conn.execute('CREATE TABLE IF NOT EXISTS '+entry_table + conn.execute('CREATE TABLE IF NOT EXISTS '+entry_table +
' (ID INTEGER PRIMARY KEY NOT NULL, Song_Id INTEGER NOT NULL, Name VARCHAR(255), Client_Id VARCHAR(36), Transferred INTEGER DEFAULT 0)') ' (ID INTEGER PRIMARY KEY NOT NULL AUTO_INCREMENT, Song_Id INTEGER NOT NULL, Name VARCHAR(255), Client_Id VARCHAR(36), Transferred INTEGER DEFAULT 0)')
conn.close()
def create_done_song_table(): def create_done_song_table():
conn = open_db() with get_db_engine().connect() as conn:
conn.execute('CREATE TABLE IF NOT EXISTS '+done_table + conn.execute('CREATE TABLE IF NOT EXISTS '+done_table +
' (Song_Id INTEGER PRIMARY KEY NOT NULL, Plays INTEGER)') ' (Song_Id INTEGER PRIMARY KEY NOT NULL, Plays INTEGER)')
conn.close()
def create_song_table(): def create_song_table():
conn = open_db() with get_db_engine().connect() as conn:
conn.execute("CREATE TABLE IF NOT EXISTS \""+song_table+"""\" ( conn.execute("CREATE TABLE IF NOT EXISTS `"+song_table+"""` (
"Id" INTEGER, `Id` INTEGER,
"Title" TEXT, `Title` TEXT,
"Artist" TEXT, `Artist` TEXT,
"Year" INTEGER, `Year` INTEGER,
"Duo" INTEGER, `Duo` INTEGER,
"Explicit" INTEGER, `Explicit` INTEGER,
"Date Added" TEXT, `Date Added` TEXT,
"Styles" TEXT, `Styles` TEXT,
"Languages" TEXT `Languages` TEXT
)""") )""")
conn.close()
def create_list_view(): def create_list_view():
conn = open_db() with get_db_engine().connect() as conn:
conn.execute("""CREATE VIEW IF NOT EXISTS [Liste] AS conn.execute("""CREATE OR REPLACE VIEW `Liste` AS
SELECT Name, Title, Artist, entries.Id, songs.Id, entries.Transferred SELECT Name, Title, Artist, entries.Id AS entry_ID, songs.Id AS song_ID, entries.Transferred
FROM entries, songs FROM entries, songs
WHERE entries.Song_Id=songs.Id""") WHERE entries.Song_Id=songs.Id""")
conn.close()
def create_done_song_view(): def create_done_song_view():
conn = open_db() with get_db_engine().connect() as conn:
conn.execute("""CREATE VIEW IF NOT EXISTS [Abspielliste] AS conn.execute("""CREATE OR REPLACE VIEW `Abspielliste` AS
SELECT Artist || \" - \" || Title AS Song, Plays AS Wiedergaben SELECT CONCAT(Artist," - ", Title) AS Song, Plays AS Wiedergaben
FROM songs, done_songs FROM songs, done_songs
WHERE done_songs.Song_Id=songs.Id""") WHERE done_songs.Song_Id=songs.Id""")
conn.close()
def get_list(): def get_list():
conn = open_db() with get_db_engine().connect() as conn:
conn.row_factory = sqlite3.Row cur = conn.execute("SELECT * FROM Liste")
cur = conn.cursor()
cur.execute("SELECT * FROM Liste")
return cur.fetchall() return cur.fetchall()
def get_played_list(): def get_played_list():
conn = open_db() with get_db_engine().connect() as conn:
cur = conn.cursor() cur = conn.execute("SELECT * FROM Abspielliste")
cur.execute("SELECT * FROM Abspielliste")
return cur.fetchall() return cur.fetchall()
def get_song_list(): def get_song_list():
conn = open_db() with get_db_engine().connect() as conn:
cur = conn.cursor() cur = conn.execute(
cur.execute("SELECT Artist || \" - \" || Title AS Song, Id FROM songs;") "SELECT Artist || \" - \" || Title AS Song, Id FROM songs;")
return cur.fetchall() return cur.fetchall()
def get_song_completions(input_string): def get_song_completions(input_string):
conn = open_db() with get_db_engine().connect() as conn:
cur = conn.cursor() # Don't look, it burns...
# Don't look, it burns... prepared_string = "%{0}%".format(
prepared_string = "%{0}%".format( input_string).upper() # "Test" -> "%TEST%"
input_string).upper() # "Test" -> "%TEST%" print(prepared_string)
print(prepared_string) cur = conn.execute(
cur.execute( "SELECT CONCAT(Artist,\" - \",Title) AS Song, Id FROM songs WHERE CONCAT(Artist,\" - \",Title) LIKE (%s) LIMIT 20;", [prepared_string])
"SELECT Title || \" - \" || Artist AS Song, Id FROM songs WHERE REPLACE(REPLACE(REPLACE(REPLACE(UPPER( SONG ),'ö','Ö'),'ü','Ü'),'ä','Ä'),'ß','') LIKE (?) LIMIT 20;", (prepared_string,))
return cur.fetchall() return cur.fetchall()
def add_entry(name, song_id, client_id): def add_entry(name, song_id, client_id):
conn = open_db() with get_db_engine().connect() as conn:
cur = conn.cursor() conn.execute(
cur.execute( "INSERT INTO entries (Song_Id,Name,Client_Id) VALUES(%s,%s,%s);", (song_id, name, client_id))
"INSERT INTO entries (Song_Id,Name,Client_Id) VALUES(?,?,?);", (song_id, name, client_id))
conn.commit()
conn.close()
return return
def add_sung_song(entry_id): def add_sung_song(entry_id):
conn = open_db() with get_db_engine().connect() as conn:
cur = conn.cursor() cur = conn.execute(
cur.execute("""SELECT Song_Id FROM entries WHERE Id=?""", (entry_id,)) """SELECT Song_Id FROM entries WHERE Id=%s""", (entry_id,))
song_id = cur.fetchone()[0] song_id = cur.fetchone()[0] # type: ignore
cur.execute("""INSERT OR REPLACE INTO done_songs (Song_Id, Plays) conn.execute("""INSERT INTO done_songs (Song_Id, Plays) VALUES("""+str(song_id)+""",1) ON DUPLICATE KEY UPDATE Plays=Plays + 1;""")
VALUES("""+str(song_id)+""", delete_entry(entry_id)
COALESCE(
(SELECT Plays FROM done_songs
WHERE Song_Id="""+str(song_id)+"), 0) + 1)"
)
conn.commit()
delete_entry(entry_id)
conn.close()
return True return True
def toggle_transferred(entry_id): def toggle_transferred(entry_id):
conn = open_db() with get_db_engine().connect() as conn:
cur = conn.cursor() cur = conn.execute(
cur.execute("SELECT Transferred FROM entries WHERE ID =?", (entry_id,)) "SELECT Transferred FROM entries WHERE ID =%s", (entry_id,))
marked = cur.fetchall()[0][0] marked = cur.fetchall()[0][0]
if(marked == 0): if(marked == 0):
cur.execute( conn.execute(
"UPDATE entries SET Transferred = 1 WHERE ID =?", (entry_id,)) "UPDATE entries SET Transferred = 1 WHERE ID =%s", (entry_id,))
else: else:
cur.execute( conn.execute(
"UPDATE entries SET Transferred = 0 WHERE ID =?", (entry_id,)) "UPDATE entries SET Transferred = 0 WHERE ID =%s", (entry_id,))
conn.commit()
conn.close()
return True return True
def check_entry_quota(client_id): def check_entry_quota(client_id):
conn = open_db() with get_db_engine().connect() as conn:
cur = conn.cursor() cur = conn.execute(
cur.execute( "SELECT Count(*) FROM entries WHERE entries.Client_Id = %s", (client_id,))
"SELECT Count(*) FROM entries WHERE entries.Client_Id = ?", (client_id,))
return cur.fetchall()[0][0] return cur.fetchall()[0][0]
def check_queue_length(): def check_queue_length():
conn = open_db() with get_db_engine().connect() as conn:
cur = conn.cursor() cur = conn.execute("SELECT Count(*) FROM entries")
cur.execute("SELECT Count(*) FROM entries")
return cur.fetchall()[0][0] return cur.fetchall()[0][0]
def clear_played_songs(): def clear_played_songs():
conn = open_db() with get_db_engine().connect() as conn:
cur = conn.cursor() conn.execute("DELETE FROM done_songs")
cur.execute("DELETE FROM done_songs")
conn.commit()
conn.close()
return True return True
def delete_entry(id): def delete_entry(id):
conn = open_db() with get_db_engine().connect() as conn:
cur = conn.cursor() conn.execute("DELETE FROM entries WHERE id=%s", (id,))
cur.execute("DELETE FROM entries WHERE id=?", (id,))
conn.commit()
conn.close()
return True return True
@ -193,20 +169,15 @@ def delete_entries(ids):
for x in ids: for x in ids:
idlist.append((x,)) idlist.append((x,))
try: try:
conn = open_db() with get_db_engine().connect() as conn:
cur = conn.cursor() cur = conn.execute("DELETE FROM entries WHERE id=%s", idlist)
cur.executemany("DELETE FROM entries WHERE id=?", idlist)
conn.commit()
conn.close()
return cur.rowcount return cur.rowcount
except sqlite3.Error as error: except Exception as error:
return -1 return -1
def delete_all_entries(): def delete_all_entries():
conn = open_db() with get_db_engine().connect() as conn:
cur = conn.cursor() conn.execute("DELETE FROM entries")
cur.execute("DELETE FROM entries")
conn.commit()
conn.close()
return True return True

View File

@ -37,7 +37,7 @@ def check_config_exists():
def load_version(app): def load_version(app):
if os.environ.get("SOURCE_VERSION"): if os.environ.get("SOURCE_VERSION"):
app.config['VERSION'] = os.environ.get("SOURCE_VERSION")[0:7] app.config['VERSION'] = os.environ.get("SOURCE_VERSION")[0:7] # type: ignore
elif os.path.isfile(".version"): elif os.path.isfile(".version"):
with open('.version', 'r') as file: with open('.version', 'r') as file:
data = file.read().replace('\n', '') data = file.read().replace('\n', '')
@ -47,22 +47,52 @@ def load_version(app):
app.config['VERSION'] = "" app.config['VERSION'] = ""
else: else:
app.config['VERSION'] = "" app.config['VERSION'] = ""
def load_dbconfig(app):
if os.environ.get("FLASK_ENV") == "development":
app.config['DBCONNSTRING'] = os.environ.get("DBSTRING")
else:
if os.environ.get("DEPLOYMENT_PLATFORM") == "Heroku":
if os.environ.get("JAWSDB_MARIA_URL"):
app.config['DBCONNSTRING'] = os.environ.get("JAWSDB_MARIA_URL")
else:
app.config['DBCONNSTRING'] = ""
if os.environ.get("DEPLOYMENT_PLATFORM") == "Docker":
if os.environ.get("DBSTRING"):
app.config['DBCONNSTRING'] = os.environ.get("DBSTRING")
else:
app.config['DBCONNSTRING'] = ""
elif os.path.isfile(".dbconn"):
with open('.dbconn', 'r') as file:
data = file.read().replace('\n', '')
if data:
app.config['DBCONNSTRING'] = data
else:
app.config['DBCONNSTRING'] = ""
else:
app.config['DBCONNSTRING'] = ""
def setup_config(app): def setup_config(app):
if check_config_exists(): if os.environ.get("DEPLOYMENT_PLATFORM") == "Docker":
config = json.load(open(config_file)) app.config['BASIC_AUTH_USERNAME'] = os.environ.get('BASIC_AUTH_USERNAME')
with open(config_file, 'r') as handle: app.config['BASIC_AUTH_PASSWORD'] = os.environ.get('BASIC_AUTH_PASSWORD')
config = json.load(handle) app.config['ENTRY_QUOTA'] = os.environ.get('ENTRY_QUOTA')
print("Loaded existing config") app.config['MAX_QUEUE'] = os.environ.get('MAX_QUEUE')
else: else:
config = {'username': 'admin', 'password': 'changeme', 'entryquota': 3, 'maxqueue': 20} if check_config_exists():
with open(config_file, 'w') as handle: config = json.load(open(config_file))
json.dump(config, handle, indent=4, sort_keys=True) with open(config_file, 'r') as handle:
print("Wrote new config") config = json.load(handle)
app.config['BASIC_AUTH_USERNAME'] = config['username'] print("Loaded existing config")
app.config['BASIC_AUTH_PASSWORD'] = config['password'] else:
app.config['ENTRY_QUOTA'] = config['entryquota'] config = {'username': 'admin', 'password': 'changeme', 'entryquota': 3, 'maxqueue': 20}
app.config['MAX_QUEUE'] = config['maxqueue'] with open(config_file, 'w') as handle:
json.dump(config, handle, indent=4, sort_keys=True)
print("Wrote new config")
app.config['BASIC_AUTH_USERNAME'] = config['username']
app.config['BASIC_AUTH_PASSWORD'] = config['password']
app.config['ENTRY_QUOTA'] = config['entryquota']
app.config['MAX_QUEUE'] = config['maxqueue']
@ -70,7 +100,7 @@ def nocache(view):
@wraps(view) @wraps(view)
def no_cache(*args, **kwargs): def no_cache(*args, **kwargs):
response = make_response(view(*args, **kwargs)) response = make_response(view(*args, **kwargs))
response.headers['Last-Modified'] = datetime.now() response.headers['Last-Modified'] = datetime.now() # type: ignore
response.headers['Cache-Control'] = 'no-store, no-cache, must-revalidate, post-check=0, pre-check=0, max-age=0' response.headers['Cache-Control'] = 'no-store, no-cache, must-revalidate, post-check=0, pre-check=0, max-age=0'
response.headers['Pragma'] = 'no-cache' response.headers['Pragma'] = 'no-cache'
response.headers['Expires'] = '-1' response.headers['Expires'] = '-1'

View File

@ -2,4 +2,6 @@ requests
pandas pandas
Flask-BasicAuth Flask-BasicAuth
bs4 bs4
gunicorn gunicorn
SQLAlchemy
mysqlclient

View File

@ -12,7 +12,7 @@
<title>{% block title %}{% endblock %} - KaraoQueue</title> <title>{% block title %}{% endblock %} - KaraoQueue</title>
<!-- Bootstrap-Tables --> <!-- Bootstrap-Tables -->
<link rel="stylesheet" href="https://unpkg.com/bootstrap-table@1.15.3/dist/bootstrap-table.min.css"> <link rel="stylesheet" href="https://unpkg.com/bootstrap-table@1.21.2/dist/bootstrap-table.min.css">
<!-- Bootstrap core CSS --> <!-- Bootstrap core CSS -->
<link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.1.3/css/bootstrap.min.css" <link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.1.3/css/bootstrap.min.css"
@ -73,9 +73,9 @@
{% if not auth %} {% if not auth %}
<a href="/login" class="ml-1 mr-1"><i class="fas fa-sign-in-alt mr-1"></i><span>Login</span></a> <a href="/login" class="ml-1 mr-1"><i class="fas fa-sign-in-alt mr-1"></i><span>Login</span></a>
{% endif %} {% endif %}
<!--<a href="https://github.com/PhoenixTwoFive/karaoqueue" <a href="https://github.com/PhoenixTwoFive/karaoqueue"
class="ml-1 mr-1"><i class="fab fa-github mr-1"></i><span>Github</span></a>--> class="ml-1 mr-1"><i class="fab fa-github mr-1"></i><span>Github</span></a>
<span class="text-muted">KaraoQueue {{karaoqueue_version}} -&nbsp;<span>&copy</span>&nbsp;2019-21 - Phillip <span class="text-muted"> {{karaoqueue_version}} -&nbsp;2019-23 - Phillip
Kühne</span> Kühne</span>
</div> </div>
</footer> </footer>
@ -96,9 +96,9 @@
<script src="https://cdnjs.cloudflare.com/ajax/libs/bootbox.js/4.4.0/bootbox.min.js" <script src="https://cdnjs.cloudflare.com/ajax/libs/bootbox.js/4.4.0/bootbox.min.js"
integrity="sha256-4F7e4JsAJyLUdpP7Q8Sah866jCOhv72zU5E8lIRER4w=" crossorigin="anonymous"> integrity="sha256-4F7e4JsAJyLUdpP7Q8Sah866jCOhv72zU5E8lIRER4w=" crossorigin="anonymous">
</script> </script>
<script src="https://unpkg.com/bootstrap-table@1.15.3/dist/bootstrap-table.min.js"></script> <script src="https://unpkg.com/bootstrap-table@1.21.2/dist/bootstrap-table.min.js"></script>
<script <script
src="https://unpkg.com/bootstrap-table@1.15.3/dist/extensions/auto-refresh/bootstrap-table-auto-refresh.min.js"></script> src="https://unpkg.com/bootstrap-table@1.21.2/dist/extensions/auto-refresh/bootstrap-table-auto-refresh.min.js"></script>
<script src="https://gitcdn.github.io/bootstrap-toggle/2.2.2/js/bootstrap-toggle.min.js"></script> <script src="https://gitcdn.github.io/bootstrap-toggle/2.2.2/js/bootstrap-toggle.min.js"></script>
{% block extrajs %}{% endblock %} {% block extrajs %}{% endblock %}
<script> <script>

View File

@ -52,10 +52,10 @@ table td:nth-child(2) {
{% block extrajs %} {% block extrajs %}
<script> <script>
$(function () { $(function () {
refreshEntryToggle()
$('#entryToggle').change(function() { $('#entryToggle').change(function() {
$.ajax({url: "/api/entries/accept/"+($('#entryToggle').is(":checked") ? "1" : "0"), complete: setTimeout(refreshEntryToggle, 1000)}); $.ajax({url: "/api/entries/accept/"+($('#entryToggle').is(":checked") ? "1" : "0"), complete: setTimeout(refreshEntryToggle, 1000)});
}) })
refreshEntryToggle()
$("#entrytable").bootstrapTable().on('load-success.bs.table', function() { $("#entrytable").bootstrapTable().on('load-success.bs.table', function() {
$('[data-toggle="tooltip"]').tooltip() $('[data-toggle="tooltip"]').tooltip()
}) })
@ -195,17 +195,19 @@ table td:nth-child(2) {
}); });
} }
function TableActions (value, row, index) { function TableActions (value, row, index) {
console.log("Value: "+value+", Row: "+row+", Index: "+index)
console.log(row)
let outerHTML = "" let outerHTML = ""
if (row.Transferred==1) { if (row.Transferred==1) {
outerHTML = "<button type=\"button\" class=\"btn btn-default\" onclick=\"markEntryAsTransferred("+row.ID+")\"><i class=\"fas fa-backward\"></i></button>&nbsp;<button type=\"button\" class=\"btn btn-success\" data-toggle=\"tooltip\" data-placement=\"top\" title=\"Als gesungen markieren\" onclick=\"markEntryAsSung("+row.ID+")\"><i class=\"fas fa-check\"></i></button>&nbsp;<button type=\"button\" class=\"btn btn-danger\" data-toggle=\"tooltip\" data-placement=\"top\" title=\"Eintrag löschen\" onclick=\"confirmDeleteEntry('"+row.Name+"',"+row.ID+")\"><i class=\"fas fa-trash\"></i></button>"; outerHTML = "<button type=\"button\" class=\"btn btn-default\" onclick=\"markEntryAsTransferred("+row.entry_ID+")\"><i class=\"fas fa-backward\"></i></button>&nbsp;<button type=\"button\" class=\"btn btn-success\" data-toggle=\"tooltip\" data-placement=\"top\" title=\"Als gesungen markieren\" onclick=\"markEntryAsSung("+row.entry_ID+")\"><i class=\"fas fa-check\"></i></button>&nbsp;<button type=\"button\" class=\"btn btn-danger\" data-toggle=\"tooltip\" data-placement=\"top\" title=\"Eintrag löschen\" onclick=\"confirmDeleteEntry('"+row.Name+"',"+row.entry_ID+")\"><i class=\"fas fa-trash\"></i></button>";
} else { } else {
outerHTML = "<button type=\"button\" class=\"btn btn-info\" onclick=\"markEntryAsTransferred("+row.ID+")\"><i class=\"fas fa-exchange-alt\"></i></button>&nbsp;<button type=\"button\" class=\"btn btn-success\" data-toggle=\"tooltip\" data-placement=\"top\" title=\"Als gesungen markieren\" onclick=\"markEntryAsSung("+row.ID+")\"><i class=\"fas fa-check\"></i></button>&nbsp;<button type=\"button\" class=\"btn btn-danger\" data-toggle=\"tooltip\" data-placement=\"top\" title=\"Eintrag löschen\" onclick=\"confirmDeleteEntry('"+row.Name+"',"+row.ID+")\"><i class=\"fas fa-trash\"></i></button>"; outerHTML = "<button type=\"button\" class=\"btn btn-info\" onclick=\"markEntryAsTransferred("+row.entry_ID+")\"><i class=\"fas fa-exchange-alt\"></i></button>&nbsp;<button type=\"button\" class=\"btn btn-success\" data-toggle=\"tooltip\" data-placement=\"top\" title=\"Als gesungen markieren\" onclick=\"markEntryAsSung("+row.entry_ID+")\"><i class=\"fas fa-check\"></i></button>&nbsp;<button type=\"button\" class=\"btn btn-danger\" data-toggle=\"tooltip\" data-placement=\"top\" title=\"Eintrag löschen\" onclick=\"confirmDeleteEntry('"+row.Name+"',"+row.entry_ID+")\"><i class=\"fas fa-trash\"></i></button>";
} }
return outerHTML; return outerHTML;
} }
function getIdSelections() { function getIdSelections() {
return $.map($("#entrytable").bootstrapTable('getSelections'), function (row) { return $.map($("#entrytable").bootstrapTable('getSelections'), function (row) {
return row.ID return row.entry_ID
}) })
} }
</script> </script>

28
docker-compose.prod.yml Normal file
View File

@ -0,0 +1,28 @@
version: "3.9"
secrets:
secrets:
file: ./secrets.yml
services:
karaoqueue:
image: "phillipkhne/karaoqueue:latest"
restart: always
ports:
- "127.0.0.1:8081:80" # Please put a reverse proxy in front of this
environment:
DEPLOYMENT_PLATFORM: Docker
DBSTRING: mysql://user:pass@host:3306/database
BASIC_AUTH_USERNAME: admin
BASIC_AUTH_PASSWORD: changeme
ENTRY_QUOTA: 3
MAX_QUEUE: 20
db:
image: mariadb
restart: always
environment:
MARIADB_ROOT_PASSWORD: changeme!
MARIADB_ROOT_HOST: localhost
MARIADB_DATABASE: karaoqueue
MARIADB_USER: karaoqueue
MARIADB_PASSWORD: change

15
docker-compose.yml Normal file
View File

@ -0,0 +1,15 @@
# This Compose file is for development only. It is not intended for production use.
# It only starts auxiliary services, such as a database, that are required for the
# application to run. The application itself is started separately, using the
# command "python -m flask run" or your favorite IDE.
# Useful for attaching a debugger to the application.
version: "3.9"
services:
db:
image: mariadb
restart: always
env_file: .env.dev
ports:
- "3306:3306"