Compare commits

...

24 Commits

Author SHA1 Message Date
8c735866a3 Remove remnants of file based config 2023-03-30 22:45:08 +02:00
035394c36b Rephrase Information text 2023-03-30 22:37:39 +02:00
1d4f77ea19 Merge pull request #49 from PhoenixTwoFive/PhoenixTwoFive/issue45
Inform user that entries are closed when big button on main page is pressed and disabled
2023-03-30 22:34:52 +02:00
573f58d764 Inform user that entries are closed when big button on main page is pressed and disabled
Fixes #45
2023-03-30 22:33:50 +02:00
dff46404f6 Merge pull request #48 from PhoenixTwoFive:bugfix/47-misaligned-buttons
Fix button alignment
2023-03-30 22:28:57 +02:00
4fd58bc39f Fix button alignment 2023-03-30 22:28:12 +02:00
bf97e9e5e4 Fix container version 2023-03-30 22:21:42 +02:00
e08041338a Add persistent volume to mariadb container. 2023-03-30 21:18:40 +02:00
4c64144f3d Fix sticky Popups 2023-03-30 21:05:36 +02:00
fad21ba6c5 Fix DELETE statements 2023-03-30 19:30:27 +02:00
3f09b79844 Fix Dockerfile for "new" WSGI server 2023-03-30 19:30:13 +02:00
1bfe3b5d4b Update Dockerfile 2023-03-30 19:14:24 +02:00
f055a59a38 Fix Dockerfile with in-Container update 2023-03-30 18:14:40 +02:00
dc53d8a8b1 Fix Typos 2023-03-30 18:00:23 +02:00
fe71fa2d8c Fix Caching 2023-03-30 18:00:08 +02:00
7ef938a5ff Create new config with credentials from env vars 2023-03-30 17:51:20 +02:00
12207c1246 Admin credentials can be changed via settings 2023-03-30 17:39:44 +02:00
16cb9e7d5a Fix theme setting persistence 2023-03-30 17:39:13 +02:00
429ffddced Update data_adapters for sqlalchemy upgrade 2023-03-30 17:38:34 +02:00
ca73d57567 Update Requirements 2023-03-30 17:07:01 +02:00
cf6d586856 Overhaul database code 2023-03-30 17:06:52 +02:00
24458a78d0 Fix bugs 2023-03-30 17:06:36 +02:00
a2cf4fc47f Change .env.dev to reflect new settings storage 2023-03-30 17:05:46 +02:00
e84ff1a381 Change container registry to github 2023-03-30 17:05:17 +02:00
13 changed files with 310 additions and 188 deletions

View File

@ -7,8 +7,6 @@ MARIADB_PASSWORD=mariadb_karaoqueue_password
# Karaoqueue # Karaoqueue
DEPLOYMENT_PLATFORM=Docker DEPLOYMENT_PLATFORM=Docker
DBSTRING=mysql://karaoqueue:mariadb_karaoqueue_password@127.0.0.1:3306/karaoqueue DBSTRING="mysql+pymysql://karaoqueue:mariadb_karaoqueue_password@127.0.0.1:3306/karaoqueue?charset=utf8mb4"
BASIC_AUTH_USERNAME=admin INITIAL_USERNAME=admin
BASIC_AUTH_PASSWORD=change_me INITIAL_PASSWORD=changeme
ENTRY_QUOTA=3
MAX_QUEUE=20

View File

@ -1,5 +1,16 @@
FROM tiangolo/uwsgi-nginx-flask:python3.10 FROM tiangolo/meinheld-gunicorn-flask:python3.9
RUN apt-key adv --recv-keys --keyserver keyserver.ubuntu.com 0xcbcb082a1bb943db
RUN curl -LsS https://downloads.mariadb.com/MariaDB/mariadb_repo_setup | bash
RUN apt-get update
RUN apt-get upgrade -y
RUN apt-get dist-upgrade
COPY ./backend/requirements.txt /app/requirements.txt
RUN pip install --no-cache-dir --upgrade -r /app/requirements.txt
RUN pip install --no-cache-dir -U meinheld
COPY ./backend /app COPY ./backend /app
RUN pip install -r /app/requirements.txt

View File

@ -83,6 +83,9 @@ def settings_post():
entryquota = request.form.get("entryquota") entryquota = request.form.get("entryquota")
maxqueue = request.form.get("maxqueue") maxqueue = request.form.get("maxqueue")
theme = request.form.get("theme") theme = request.form.get("theme")
username = request.form.get("username")
password = request.form.get("password")
changed_credentials = False
if entryquota.isnumeric() and int(entryquota) > 0: # type: ignore if entryquota.isnumeric() and int(entryquota) > 0: # type: ignore
app.config['ENTRY_QUOTA'] = int(entryquota) # type: ignore app.config['ENTRY_QUOTA'] = int(entryquota) # type: ignore
else: else:
@ -91,14 +94,21 @@ def settings_post():
app.config['MAX_QUEUE'] = int(maxqueue) # type: ignore app.config['MAX_QUEUE'] = int(maxqueue) # type: ignore
else: else:
abort(400) abort(400)
if theme in helpers.get_themes(): if theme is not None and theme in helpers.get_themes():
app.config['THEME'] = theme helpers.set_theme(app,theme)
else: else:
abort(400) abort(400)
if username != "" and username != app.config['BASIC_AUTH_USERNAME']:
app.config['BASIC_AUTH_USERNAME'] = username
changed_credentials = True
if password != "":
app.config['BASIC_AUTH_PASSWORD'] = password
changed_credentials = True
helpers.persist_config(app=app) helpers.persist_config(app=app)
if changed_credentials:
return render_template('settings.html', app=app, auth=basic_auth.authenticate()) return redirect("/")
else:
return render_template('settings.html', app=app, auth=basic_auth.authenticate(), themes=helpers.get_themes())
@app.route("/api/queue") @app.route("/api/queue")
@ -133,7 +143,7 @@ def update_songs():
return Response('{"status": "%s" }' % status, mimetype='text/json') return Response('{"status": "%s" }' % status, mimetype='text/json')
@app.route("/api/songs/compl") @app.route("/api/songs/compl") # type: ignore
@nocache @nocache
def get_song_completions(input_string=""): def get_song_completions(input_string=""):
input_string = request.args.get('search', input_string) input_string = request.args.get('search', input_string)
@ -238,7 +248,6 @@ def admin():
def activate_job(): def activate_job():
helpers.load_dbconfig(app) helpers.load_dbconfig(app)
helpers.load_version(app) helpers.load_version(app)
helpers.create_data_directory()
database.create_entry_table() database.create_entry_table()
database.create_song_table() database.create_song_table()
database.create_done_song_table() database.create_done_song_table()
@ -256,7 +265,7 @@ def add_header(response):
and also to cache the rendered page for 10 minutes. and also to cache the rendered page for 10 minutes.
""" """
if not 'Cache-Control' in response.headers: if not 'Cache-Control' in response.headers:
response.headers['Cache-Control'] = 'private, max-age=600' response.headers['Cache-Control'] = 'private, max-age=600, no-cache, must-revalidate'
return response return response
@app.context_processor @app.context_processor

View File

@ -1,8 +1,5 @@
def dict_from_row(row):
return dict(zip(row.keys(), row))
def dict_from_rows(rows): def dict_from_rows(rows):
outlist=[] outlist=[]
for row in rows: for row in rows:
outlist.append(dict_from_row(row)) outlist.append(dict(row._mapping))
return outlist return outlist

View File

@ -2,7 +2,7 @@
from email.mime import base from email.mime import base
from MySQLdb import Connection from MySQLdb import Connection
from sqlalchemy import create_engine, engine from sqlalchemy import create_engine, engine, text
import pandas import pandas
from io import StringIO from io import StringIO
from flask import current_app from flask import current_app
@ -19,7 +19,8 @@ def get_db_engine() -> engine.base.Engine:
global sql_engine global sql_engine
if (not sql_engine): if (not sql_engine):
print(current_app.config.get("DBCONNSTRING")) print(current_app.config.get("DBCONNSTRING"))
sql_engine = create_engine(current_app.config.get("DBCONNSTRING")) # type: ignore sql_engine = create_engine(
current_app.config.get("DBCONNSTRING")) # type: ignore
return sql_engine return sql_engine
@ -28,147 +29,171 @@ def import_songs(song_csv):
df = pandas.read_csv(StringIO(song_csv), sep=';') df = pandas.read_csv(StringIO(song_csv), sep=';')
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
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 = conn.execute(text("SELECT Count(Id) FROM songs"))
num_songs = cur.fetchone()[0] # type: ignore num_songs = cur.fetchone()[0] # type: ignore
conn.commit()
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():
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
conn.execute('CREATE TABLE IF NOT EXISTS '+entry_table + stmt = text(
' (ID INTEGER PRIMARY KEY NOT NULL AUTO_INCREMENT, Song_Id INTEGER NOT NULL, Name VARCHAR(255), Client_Id VARCHAR(36), Transferred INTEGER DEFAULT 0)') f'CREATE TABLE IF NOT EXISTS `{entry_table}` (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.execute(stmt)
conn.commit()
def create_done_song_table(): def create_done_song_table():
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
conn.execute('CREATE TABLE IF NOT EXISTS '+done_table + stmt = text(
' (Song_Id INTEGER PRIMARY KEY NOT NULL, Plays INTEGER)') f'CREATE TABLE IF NOT EXISTS `{done_table}` (Song_Id INTEGER PRIMARY KEY NOT NULL, Plays INTEGER)')
conn.execute(stmt)
conn.commit()
def create_song_table(): def create_song_table():
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
conn.execute("CREATE TABLE IF NOT EXISTS `"+song_table+"""` ( stmt = text(f"""CREATE TABLE IF NOT EXISTS `{song_table}` (
`Id` INTEGER, `Id` INTEGER,
`Title` TEXT, `Title` TEXT,
`Artist` TEXT, `Artist` TEXT,
`Year` INTEGER, `Year` VARCHAR(4),
`Duo` INTEGER, `Duo` BOOLEAN,
`Explicit` INTEGER, `Explicit` INTEGER,
`Date Added` TEXT, `Date Added` TIMESTAMP,
`Styles` TEXT, `Styles` TEXT,
`Languages` TEXT `Languages` TEXT
)""") )""")
conn.execute(stmt)
conn.commit()
def create_list_view(): def create_list_view():
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
conn.execute("""CREATE OR REPLACE VIEW `Liste` AS stmt = text("""CREATE OR REPLACE VIEW `Liste` AS
SELECT Name, Title, Artist, entries.Id AS entry_ID, songs.Id AS song_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.execute(stmt)
conn.commit()
def create_done_song_view(): def create_done_song_view():
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
conn.execute("""CREATE OR REPLACE VIEW `Abspielliste` AS stmt = text("""CREATE OR REPLACE VIEW `Abspielliste` AS
SELECT CONCAT(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.execute(stmt)
conn.commit()
def create_config_table(): def create_config_table():
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
conn.execute("""CREATE TABLE IF NOT EXISTS `config` ( stmt = text("""CREATE TABLE IF NOT EXISTS `config` (
`Key` VARCHAR(50) NOT NULL PRIMARY KEY, `Key` VARCHAR(50) NOT NULL PRIMARY KEY,
`Value` TEXT `Value` TEXT
)""") )""")
conn.execute(stmt)
conn.commit()
def get_list(): def get_list():
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
cur = conn.execute("SELECT * FROM Liste") stmt = text("SELECT * FROM Liste")
cur = conn.execute(stmt)
return cur.fetchall() return cur.fetchall()
def get_played_list(): def get_played_list():
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
cur = conn.execute("SELECT * FROM Abspielliste") stmt = text("SELECT * FROM Abspielliste")
cur = conn.execute(stmt)
return cur.fetchall() return cur.fetchall()
def get_song_list(): def get_song_list():
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
cur = conn.execute( stmt = text("SELECT Artist || \" - \" || Title AS Song, Id FROM songs;")
"SELECT Artist || \" - \" || Title AS Song, Id FROM songs;") cur = conn.execute(stmt)
return cur.fetchall() return cur.fetchall()
def get_song_completions(input_string): def get_song_completions(input_string):
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
# Don't look, it burns... prepared_string = f"%{input_string.upper()}%"
prepared_string = "%{0}%".format( stmt = text(
input_string).upper() # "Test" -> "%TEST%" "SELECT CONCAT(Artist, ' - ', Title) AS Song, Id FROM songs WHERE CONCAT(Artist, ' - ', Title) LIKE :prepared_string LIMIT 20;")
print(prepared_string)
cur = conn.execute( cur = conn.execute(
"SELECT CONCAT(Artist,\" - \",Title) AS Song, Id FROM songs WHERE CONCAT(Artist,\" - \",Title) LIKE (%s) LIMIT 20;", [prepared_string]) stmt, {"prepared_string": prepared_string}) # type: ignore
return cur.fetchall() return cur.fetchall()
def add_entry(name, song_id, client_id): def add_entry(name, song_id, client_id):
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
conn.execute( stmt = text(
"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 (:par_song_id,:par_name,:par_client_id);")
return conn.execute(stmt, {"par_song_id": song_id, "par_name": name,
"par_client_id": client_id}) # type: ignore
conn.commit()
return True
def add_sung_song(entry_id): def add_sung_song(entry_id):
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
cur = conn.execute( stmt = text("SELECT Song_Id FROM entries WHERE Id=:par_entry_id")
"""SELECT Song_Id FROM entries WHERE Id=%s""", (entry_id,)) cur = conn.execute(stmt, {"par_entry_id": entry_id}) # type: ignore
song_id = cur.fetchone()[0] # type: ignore song_id = cur.fetchone()[0] # type: ignore
conn.execute("""INSERT INTO done_songs (Song_Id, Plays) VALUES("""+str(song_id)+""",1) ON DUPLICATE KEY UPDATE Plays=Plays + 1;""") stmt = text(
"INSERT INTO done_songs (Song_Id,Plays) VALUES (:par_song_id,1) ON DUPLICATE KEY UPDATE Plays=Plays + 1;")
conn.execute(stmt, {"par_song_id": song_id}) # type: ignore
conn.commit()
delete_entry(entry_id) delete_entry(entry_id)
return True return True
def toggle_transferred(entry_id): def toggle_transferred(entry_id):
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
cur = conn.execute( cur = conn.execute(text("SELECT Transferred FROM entries WHERE ID = :par_entry_id"),
"SELECT Transferred FROM entries WHERE ID =%s", (entry_id,)) {"par_entry_id": entry_id}) # type: ignore
marked = cur.fetchall()[0][0] marked = cur.fetchall()[0][0]
if(marked == 0): if (marked == 0):
conn.execute( conn.execute(text("UPDATE entries SET Transferred = 1 WHERE ID = :par_entry_id"),
"UPDATE entries SET Transferred = 1 WHERE ID =%s", (entry_id,)) {"par_entry_id": entry_id}) # type: ignore
else: else:
conn.execute( conn.execute(text("UPDATE entries SET Transferred = 0 WHERE ID = :par_entry_id"),
"UPDATE entries SET Transferred = 0 WHERE ID =%s", (entry_id,)) {"par_entry_id": entry_id}) # type: ignore
conn.commit()
return True return True
def check_entry_quota(client_id): def check_entry_quota(client_id):
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
cur = conn.execute( cur = conn.execute(text("SELECT Count(*) FROM entries WHERE entries.Client_Id = :par_client_id"),
"SELECT Count(*) FROM entries WHERE entries.Client_Id = %s", (client_id,)) {"par_client_id": client_id}) # type: ignore
return cur.fetchall()[0][0] return cur.fetchall()[0][0]
def check_queue_length(): def check_queue_length():
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
cur = conn.execute("SELECT Count(*) FROM entries") cur = conn.execute(text("SELECT Count(*) FROM entries"))
return cur.fetchall()[0][0] return cur.fetchall()[0][0]
def clear_played_songs(): def clear_played_songs():
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
conn.execute("DELETE FROM done_songs") conn.execute(text("DELETE FROM done_songs"))
conn.commit()
return True return True
def delete_entry(id): def delete_entry(id):
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
conn.execute("DELETE FROM entries WHERE id=%s", (id,)) conn.execute(text("DELETE FROM entries WHERE id= :par_id"), {
"par_id": id}) # type: ignore
conn.commit()
return True return True
@ -178,8 +203,9 @@ def delete_entries(ids):
idlist.append((x,)) idlist.append((x,))
try: try:
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
cur = conn.execute("DELETE FROM entries WHERE id=%s", idlist) cur = conn.execute(text("DELETE FROM entries WHERE id= :par_id"), {
"par_id": idlist})
conn.commit()
return cur.rowcount return cur.rowcount
except Exception as error: except Exception as error:
return -1 return -1
@ -187,23 +213,50 @@ def delete_entries(ids):
def delete_all_entries() -> bool: def delete_all_entries() -> bool:
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
conn.execute("DELETE FROM entries") conn.execute(text("DELETE FROM entries"))
conn.commit()
return True return True
def get_config(key: str) -> str: def get_config(key: str) -> str:
with get_db_engine().connect() as conn: try:
cur = conn.execute("SELECT `Value` FROM config WHERE `Key`=%s", (key,)) with get_db_engine().connect() as conn:
return cur.fetchall()[0][0] cur = conn.execute(
text("SELECT `Value` FROM config WHERE `Key`= :par_key"), {"par_key": key}) # type: ignore
conn.commit()
return cur.fetchall()[0][0]
except IndexError as error:
return ""
def set_config(key: str, value: str) -> bool: def set_config(key: str, value: str) -> bool:
print(f"Setting config {key} to {value}")
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
conn.execute("INSERT INTO config (`Key`, `Value`) VALUES (%s,%s) ON DUPLICATE KEY UPDATE `Value`=%s", (key, value, value)) conn.execute(text(
"INSERT INTO config (`Key`, `Value`) VALUES ( :par_key , :par_value) ON DUPLICATE KEY UPDATE `Value`= :par_value"),
{"par_key": key, "par_value": value}
) # type: ignore
conn.commit()
return True return True
def get_config_list() -> dict: def get_config_list() -> dict:
with get_db_engine().connect() as conn: with get_db_engine().connect() as conn:
cur = conn.execute("SELECT * FROM config") cur = conn.execute(text("SELECT * FROM config"))
result_dict = {} result_dict = {}
for row in cur.fetchall(): for row in cur.fetchall():
result_dict[row[0]] = row[1] result_dict[row[0]] = row[1]
return result_dict return result_dict
def check_config_table() -> bool:
with get_db_engine().connect() as conn:
if conn.dialect.has_table(conn, 'config'):
# type: ignore
# type: ignore
if (conn.execute(text("SELECT COUNT(*) FROM config")).fetchone()[0] > 0): # type: ignore
return True
else:
return False
else:
return False

View File

@ -8,24 +8,19 @@ from functools import wraps, update_wrapper
from datetime import datetime from datetime import datetime
import database import database
data_directory = "data"
config_file = data_directory+"/config.json"
def create_data_directory():
if not os.path.exists(data_directory):
os.makedirs(data_directory)
def get_catalog_url(): def get_catalog_url():
r = requests.get('https://www.karafun.de/karaoke-song-list.html') r = requests.get('https://www.karafun.de/karaoke-song-list.html')
soup = BeautifulSoup(r.content, 'html.parser') soup = BeautifulSoup(r.content, 'html.parser')
url = soup.findAll('a', href=True, text='Verfügbar in CSV-Format')[0]['href'] url = soup.findAll(
'a', href=True, text='Verfügbar in CSV-Format')[0]['href']
return url return url
def get_songs(url): def get_songs(url):
r = requests.get(url) r = requests.get(url)
return r.text return r.text
def is_valid_uuid(val): def is_valid_uuid(val):
try: try:
uuid.UUID(str(val)) uuid.UUID(str(val))
@ -33,20 +28,14 @@ def is_valid_uuid(val):
except ValueError: except ValueError:
return False return False
def check_config_exists(): def check_config_exists():
eng = database.get_db_engine() return database.check_config_table()
with eng.connect() as conn:
if conn.dialect.has_table(conn, 'config'):
if (conn.execute("SELECT COUNT(*) FROM config").fetchone()[0] > 0): # type: ignore
return True
else:
return False
else:
return False
def load_version(app: Flask): def load_version(app: Flask):
if os.environ.get("SOURCE_VERSION"): if os.environ.get("SOURCE_VERSION"):
app.config['VERSION'] = os.environ.get("SOURCE_VERSION")[0:7] # type: ignore app.config['VERSION'] = os.environ.get("SOURCE_VERSION")[0:7] # type: ignore # noqa: E501
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', '')
@ -57,6 +46,7 @@ def load_version(app: Flask):
else: else:
app.config['VERSION'] = "" app.config['VERSION'] = ""
def load_dbconfig(app: Flask): def load_dbconfig(app: Flask):
if os.environ.get("FLASK_ENV") == "development": if os.environ.get("FLASK_ENV") == "development":
app.config['DBCONNSTRING'] = os.environ.get("DBSTRING") app.config['DBCONNSTRING'] = os.environ.get("DBSTRING")
@ -82,22 +72,41 @@ def load_dbconfig(app: Flask):
exit("No database connection string found. Cannot continue. Please set the environment variable DBSTRING or create a file .dbconn in the root directory of the project.") exit("No database connection string found. Cannot continue. Please set the environment variable DBSTRING or create a file .dbconn in the root directory of the project.")
# Check if config exists in DB, if not, create it. # Check if config exists in DB, if not, create it.
def setup_config(app: Flask): def setup_config(app: Flask):
if check_config_exists(): if check_config_exists() == False:
config = database.get_config_list() print("No config found, creating new config")
print("Loaded existing config") initial_username = os.environ.get("INITIAL_USERNAME")
else: initial_password = os.environ.get("INITIAL_PASSWORD")
config = {'username': 'admin', 'password': 'changeme', 'entryquota': 3, 'maxqueue': 20, 'entries_allowed': 1, 'theme': 'default'} if initial_username is None:
for key, value in config.items(): print(
"No initial username set. Please set the environment variable INITIAL_USERNAME")
exit()
if initial_password is None:
print(
"No initial password set. Please set the environment variable INITIAL_PASSWORD")
exit()
default_config = {'username': initial_username,
'password': initial_password,
'entryquota': 3,
'maxqueue': 20,
'entries_allowed': 1,
'theme': 'default.css'}
for key, value in default_config.items():
database.set_config(key, value) database.set_config(key, value)
print("Created new config") print("Created new config")
config = database.get_config_list()
app.config['BASIC_AUTH_USERNAME'] = config['username'] app.config['BASIC_AUTH_USERNAME'] = config['username']
app.config['BASIC_AUTH_PASSWORD'] = config['password'] app.config['BASIC_AUTH_PASSWORD'] = config['password']
app.config['ENTRY_QUOTA'] = config['entryquota'] app.config['ENTRY_QUOTA'] = config['entryquota']
app.config['MAX_QUEUE'] = config['maxqueue'] app.config['MAX_QUEUE'] = config['maxqueue']
app.config['ENTRIES_ALLOWED'] = bool(config['entries_allowed']) app.config['ENTRIES_ALLOWED'] = bool(config['entries_allowed'])
app.config['THEME'] = config['theme']
# set queue admittance # set queue admittance
def set_accept_entries(app: Flask, allowed: bool): def set_accept_entries(app: Flask, allowed: bool):
if allowed: if allowed:
app.config['ENTRIES_ALLOWED'] = True app.config['ENTRIES_ALLOWED'] = True
@ -107,6 +116,8 @@ def set_accept_entries(app: Flask, allowed: bool):
database.set_config('entries_allowed', '0') database.set_config('entries_allowed', '0')
# get queue admittance # get queue admittance
def get_accept_entries(app: Flask) -> bool: def get_accept_entries(app: Flask) -> bool:
state = bool(int(database.get_config('entries_allowed'))) state = bool(int(database.get_config('entries_allowed')))
app.config['ENTRIES_ALLOWED'] = state app.config['ENTRIES_ALLOWED'] = state
@ -115,11 +126,14 @@ def get_accept_entries(app: Flask) -> bool:
# Write settings from current app.config to DB # Write settings from current app.config to DB
def persist_config(app: Flask): def persist_config(app: Flask):
config = {'username': app.config['BASIC_AUTH_USERNAME'], 'password': app.config['BASIC_AUTH_PASSWORD'], 'entryquota': app.config['ENTRY_QUOTA'], 'maxqueue': app.config['MAX_QUEUE']} config = {'username': app.config['BASIC_AUTH_USERNAME'], 'password': app.config['BASIC_AUTH_PASSWORD'],
'entryquota': app.config['ENTRY_QUOTA'], 'maxqueue': app.config['MAX_QUEUE']}
for key, value in config.items(): for key, value in config.items():
database.set_config(key, value) database.set_config(key, value)
# Get available themes from themes directory # Get available themes from themes directory
def get_themes(): def get_themes():
themes = [] themes = []
for theme in os.listdir('./static/css/themes'): for theme in os.listdir('./static/css/themes'):
@ -127,6 +141,8 @@ def get_themes():
return themes return themes
# Set theme # Set theme
def set_theme(app: Flask, theme: str): def set_theme(app: Flask, theme: str):
if theme in get_themes(): if theme in get_themes():
app.config['THEME'] = theme app.config['THEME'] = theme

View File

@ -1,7 +1,30 @@
requests autopep8
pandas beautifulsoup4
Flask-BasicAuth
bs4 bs4
certifi
charset-normalizer
click
Flask
Flask-BasicAuth
greenlet
gunicorn gunicorn
SQLAlchemy idna
itsdangerous
Jinja2
mariadb
MarkupSafe
mysql
mysqlclient mysqlclient
numpy
pandas
pycodestyle
PyMySQL
python-dateutil
pytz
requests
six
soupsieve
SQLAlchemy
toml
urllib3
Werkzeug

View File

@ -115,7 +115,7 @@ body {
} }
.footer { .footer {
background-color: #232323; background-color: var(--background-color-var);
} }
.modal-content { .modal-content {
@ -154,6 +154,10 @@ body {
border-color: var(--table-border-color) border-color: var(--table-border-color)
} }
table td.buttoncell {
text-align: end;
}
.close { .close {
color: var(--text-color) color: var(--text-color)
} }

View File

@ -29,6 +29,9 @@ $.getJSON("/api/entries/accept", (data) => {
$("#bfb").addClass("disabled") $("#bfb").addClass("disabled")
$("#bfb").prop("aria-disabled",true); $("#bfb").prop("aria-disabled",true);
$("#bfb").prop("tabindex","-1"); $("#bfb").prop("tabindex","-1");
$("#bfb").wrap("<span class='tooltip-span' tabindex='0' data-toggle='tooltip' data-placement='bottom'></span>");
$(".tooltip-span").prop("title", "Eintragungen sind leider momentan nicht möglich.")
$('[data-toggle="tooltip"]').tooltip()
} }
}) })
</script> </script>

View File

@ -1,15 +1,13 @@
{% extends 'base.html' %} {% extends 'base.html' %}
{% block title %}Warteliste-Admin{% endblock %} {% block title %}Warteliste-Admin{% endblock %}
{% block content %} {% block content %}
<style> <style>
table td:nth-child(2) { table td:nth-child(2) {
overflow-y: hidden; overflow-y: hidden;
overflow-x: auto; overflow-x: auto;
text-overflow: clip; text-overflow: clip;
max-width: 200px !important; max-width: 200px !important;
} }
</style> </style>
<div class="container"> <div class="container">
<div id="toolbar"> <div id="toolbar">
@ -18,24 +16,13 @@ table td:nth-child(2) {
<button type="button" class="topbutton btn btn-danger" onclick="confirmUpdateSongDatabase()"><i <button type="button" class="topbutton btn btn-danger" onclick="confirmUpdateSongDatabase()"><i
class="fas fa-file-import mr-2"></i>Song-Datenbank class="fas fa-file-import mr-2"></i>Song-Datenbank
aktualisieren</button> aktualisieren</button>
<input id="entryToggle" type="checkbox" class="topbutton" data-toggle="toggle" data-on="Eintragen erlaubt" data-off="Eintragen deaktiviert" data-onstyle="success" data-offstyle="danger"> <input id="entryToggle" type="checkbox" class="topbutton" data-toggle="toggle" data-on="Eintragen erlaubt"
data-off="Eintragen deaktiviert" data-onstyle="success" data-offstyle="danger">
</div> </div>
<table class="table entries" <table class="table entries" id="entrytable" data-toggle="table" data-search="true" data-show-columns="true"
id="entrytable" data-show-toggle="true" data-multiple-select-row="true" data-click-to-select="true" data-toolbar="#toolbar"
data-toggle="table" data-pagination="true" data-show-extended-pagination="true" data-classes="table table-hover"
data-search="true" data-url="/api/queue" data-show-refresh="true" data-auto-refresh="true" data-auto-refresh-interval="10">
data-show-columns="true"
data-show-toggle="true"
data-multiple-select-row="true"
data-click-to-select="true"
data-toolbar="#toolbar"
data-pagination="true"
data-show-extended-pagination="true"
data-classes="table table-hover"
data-url="/api/queue"
data-show-refresh="true"
data-auto-refresh="true"
data-auto-refresh-interval="10">
<thead> <thead>
<tr> <tr>
<th data-field="state" data-checkbox="true"></th> <th data-field="state" data-checkbox="true"></th>
@ -53,15 +40,18 @@ table td:nth-child(2) {
<script> <script>
$(function () { $(function () {
refreshEntryToggle() 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) });
}) })
$("#entrytable").bootstrapTable().on('load-success.bs.table', function() { $("#entrytable").bootstrapTable().on('load-success.bs.table', function () {
$('[data-toggle="tooltip"]').tooltip() $('[data-toggle="tooltip"]').tooltip()
}) })
$('[data-toggle="tooltip"]').tooltip({
trigger: 'hover'
})
}) })
function confirmDeleteEntry(name, entry_id) { function confirmDeleteEntry(name, entry_id) {
bootbox.confirm("Wirklich den Eintrag von "+name+" löschen?", function(result){ bootbox.confirm("Wirklich den Eintrag von " + name + " löschen?", function (result) {
if (result) { if (result) {
deleteEntry(entry_id) deleteEntry(entry_id)
} }
@ -80,7 +70,7 @@ table td:nth-child(2) {
className: 'btn btn-secondary' className: 'btn btn-secondary'
} }
}, },
callback: function(result){ callback: function (result) {
if (result) { if (result) {
DeleteSelectedEntries(getIdSelections()) DeleteSelectedEntries(getIdSelections())
} }
@ -100,7 +90,7 @@ table td:nth-child(2) {
className: 'btn btn-secondary' className: 'btn btn-secondary'
} }
}, },
callback: function(result){ callback: function (result) {
if (result) { if (result) {
var dialog = bootbox.dialog({ var dialog = bootbox.dialog({
message: '<p class="text-center mb-0"><i class="fa fa-spin fa-cog"></i> Aktualisiere Song-Datenbank...</p>', message: '<p class="text-center mb-0"><i class="fa fa-spin fa-cog"></i> Aktualisiere Song-Datenbank...</p>',
@ -114,8 +104,8 @@ table td:nth-child(2) {
} }
function refreshEntryToggle() { function refreshEntryToggle() {
$.getJSON("/api/entries/accept", (data) => { $.getJSON("/api/entries/accept", (data) => {
if (data["value"]!=$('#entryToggle').is(":checked")) { if (data["value"] != $('#entryToggle').is(":checked")) {
if(data["value"]==1) { if (data["value"] == 1) {
$('#entryToggle').data('bs.toggle').on('true') $('#entryToggle').data('bs.toggle').on('true')
} }
else { else {
@ -125,51 +115,51 @@ table td:nth-child(2) {
}) })
} }
function deleteEntry(entry_id) { function deleteEntry(entry_id) {
$.ajax({ $.ajax({
type: 'GET', type: 'GET',
url: '/api/entries/delete/'+entry_id, url: '/api/entries/delete/' + entry_id,
contentType: "application/json", contentType: "application/json",
dataType: 'json', dataType: 'json',
async: false async: false
}); });
$("#entrytable").bootstrapTable('refresh') $("#entrytable").bootstrapTable('refresh')
} }
function markEntryAsSung(entry_id) { function markEntryAsSung(entry_id) {
$.ajax({ $.ajax({
type: 'GET', type: 'GET',
url: '/api/entries/mark_sung/'+entry_id, url: '/api/entries/mark_sung/' + entry_id,
contentType: "application/json", contentType: "application/json",
dataType: 'json', dataType: 'json',
async: false async: false
}); });
$("#entrytable").bootstrapTable('refresh') $("#entrytable").bootstrapTable('refresh')
} }
function markEntryAsTransferred(entry_id) { function markEntryAsTransferred(entry_id) {
$.ajax({ $.ajax({
type: 'GET', type: 'GET',
url: '/api/entries/mark_transferred/'+entry_id, url: '/api/entries/mark_transferred/' + entry_id,
contentType: "application/json", contentType: "application/json",
dataType: 'json', dataType: 'json',
async: false async: false
}); });
$("#entrytable").bootstrapTable('refresh') $("#entrytable").bootstrapTable('refresh')
} }
function DeleteSelectedEntries(ids) { function DeleteSelectedEntries(ids) {
$.ajax({ $.ajax({
type: 'POST', type: 'POST',
url: '/api/entries/delete', url: '/api/entries/delete',
data: JSON.stringify(ids), // or JSON.stringify ({name: 'jonas'}), data: JSON.stringify(ids), // or JSON.stringify ({name: 'jonas'}),
error: function() { error: function () {
bootbox.alert({ bootbox.alert({
message: "Fehler beim Löschen der Eintragungen.", message: "Fehler beim Löschen der Eintragungen.",
}) })
}, },
success: function() { success: function () {
$("#entrytable").bootstrapTable('refresh') $("#entrytable").bootstrapTable('refresh')
}, },
@ -183,28 +173,29 @@ table td:nth-child(2) {
url: '/api/songs/update', url: '/api/songs/update',
contentType: "application/json", contentType: "application/json",
dataType: 'json', dataType: 'json',
success: function(data) { success: function (data) {
wait_dialog.modal('hide') wait_dialog.modal('hide')
bootbox.alert({ bootbox.alert({
message: data["status"], message: data["status"],
callback: function() { callback: function () {
$("#entrytable").bootstrapTable('refresh') $("#entrytable").bootstrapTable('refresh')
} }
}) })
} }
}); });
} }
function TableActions (value, row, index) { function TableActions(value, row, index) {
console.log("Value: "+value+", Row: "+row+", Index: "+index) console.log("Value: " + value + ", Row: " + row + ", Index: " + index)
console.log(row) 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.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>"; outerHTML = "<button type=\"button\" class=\"btn btn-default\" data-toggle=\"tooltip\" data-placement=\"top\" title=\"Markierung zurückziehen\" onclick=\"event.stopPropagation();$(this).tooltip('dispose');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=\"event.stopPropagation();$(this).tooltip('dispose');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=\"event.stopPropagation();$(this).tooltip('dispose');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.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>"; outerHTML = "<button type=\"button\" class=\"btn btn-info\" data-toggle=\"tooltip\" data-placement=\"top\" title=\"Als übertragen markieren\" onclick=\"event.stopPropagation();$(this).tooltip('dispose');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=\"event.stopPropagation();$(this).tooltip('dispose');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=\"event.stopPropagation();$(this).tooltip('dispose');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.entry_ID return row.entry_ID

View File

@ -18,6 +18,18 @@
{% endfor %} {% endfor %}
</select> </select>
</p> </p>
<div class="alert alert-warning" role="alert">
<i class="fas fa-exclamation-triangle mr-1"></i>
<strong>Warnung:</strong> Änderungen an den folgenden Einstellungen führen zu einer sofortigen Abmeldung!
</div>
<p>
<label for="username">Benutzername</label>
<input type="text" class="form-control" id="username" name="username" value={{app.config['BASIC_AUTH_USERNAME']}}>
</p>
<p>
<label for="password">Passwort ändern</label>
<input type="password" class="form-control" id="password" name="password">
</p>
<input type="submit" class="btn btn-primary mr-1 mb-2" value="Einstellungen anwenden"> <input type="submit" class="btn btn-primary mr-1 mb-2" value="Einstellungen anwenden">
</form> </form>
<details> <details>

View File

@ -118,7 +118,7 @@
$(".enqueueButton").prop("disabled", true) $(".enqueueButton").prop("disabled", true)
$(".enqueueButton").prop("style", "pointer-events: none;") $(".enqueueButton").prop("style", "pointer-events: none;")
$(".enqueueButton").wrap("<span class='tooltip-span' tabindex='0' data-toggle='tooltip' data-placement='top'></span>"); $(".enqueueButton").wrap("<span class='tooltip-span' tabindex='0' data-toggle='tooltip' data-placement='top'></span>");
$(".tooltip-span").prop("title", "Eintragungen sind leider nicht mehr möglich.") $(".tooltip-span").prop("title", "Eintragungen sind leider momentan nicht möglich.")
$('[data-toggle="tooltip"]').tooltip() $('[data-toggle="tooltip"]').tooltip()
} else { } else {
$(".enqueueButton").prop("disabled", false) $(".enqueueButton").prop("disabled", false)

View File

@ -6,7 +6,7 @@ secrets:
services: services:
karaoqueue: karaoqueue:
image: "phillipkhne/karaoqueue:latest" image: "ghcr.io/phoenixtwofive/karaoqueue:v2023.03.1"
build: . build: .
restart: always restart: always
ports: ports:
@ -16,3 +16,8 @@ services:
image: mariadb image: mariadb
restart: always restart: always
env_file: .env env_file: .env
volumes:
- karaoqueue-db:/var/lib/mysql
volumes:
karaoqueue-db: