Many fixes and improvements
Fixes: #20,#22,#23,#24,#25,#26,#27,#28,#29,#30,#31,#32,#33,#34,#35,#36,#37,#38,#39,#40,#41,#42,#43,#46,#48
This commit is contained in:
parent
d263701297
commit
78ed396c36
81
flask/app.py
81
flask/app.py
|
@ -13,12 +13,16 @@ db.startup()
|
||||||
app = Flask(__name__)
|
app = Flask(__name__)
|
||||||
|
|
||||||
########## CUSTOM FILTERS
|
########## CUSTOM FILTERS
|
||||||
|
## converts a unix timestamp to a human-readable format
|
||||||
|
## OUTPUT: str="2023-11-02 12:33"
|
||||||
@app.template_filter('ctime')
|
@app.template_filter('ctime')
|
||||||
def timectime(s):
|
def timectime(s:int):
|
||||||
return datetime.utcfromtimestamp(s).strftime('%Y-%m-%d %H:%M')
|
return datetime.utcfromtimestamp(s).strftime('%Y-%m-%d %H:%M')
|
||||||
|
|
||||||
|
## converts a bytes to a human readable scale
|
||||||
|
## OUTPUT: str="XXX.XXYB"
|
||||||
@app.template_filter('spacer')
|
@app.template_filter('spacer')
|
||||||
def convsize(s):
|
def convsize(s:int):
|
||||||
sizes=("B","KB","MB","GB","TB")
|
sizes=("B","KB","MB","GB","TB")
|
||||||
n=0
|
n=0
|
||||||
while s >= 1000:
|
while s >= 1000:
|
||||||
|
@ -31,19 +35,21 @@ def convsize(s):
|
||||||
@app.route('/')
|
@app.route('/')
|
||||||
def homepage():
|
def homepage():
|
||||||
# try to get userdata, else logout state
|
# try to get userdata, else logout state
|
||||||
logged_in,userdata=get_login_info(request.cookies.get('session'))
|
logged_in,userdata=get_login_info(request.cookies.get('session-id'))
|
||||||
archives=db.get_n_archives()
|
archives=db.get_n_archives()
|
||||||
return render_template("home.html", title="Homepage",userdata=userdata,login=logged_in,archives=archives)
|
return render_template("home.html", title="Homepage",userdata=userdata,login=logged_in,archives=archives)
|
||||||
|
|
||||||
@app.route('/user')
|
@app.route('/user')
|
||||||
@app.route('/user/<int:page_userid>', methods=['GET','POST'])
|
@app.route('/user/<int:page_userid>', methods=['GET','POST'])
|
||||||
def userpage(page_userid:int=0):
|
def userpage(page_userid:int=0):
|
||||||
logged_in,userdata=get_login_info(request.cookies.get('session'))
|
logged_in,userdata=get_login_info(request.cookies.get('session-id'))
|
||||||
if not logged_in:
|
if not logged_in:
|
||||||
return make_response(redirect('/'))
|
return make_response(redirect('/'))
|
||||||
if page_userid == 0:
|
if page_userid == 0:
|
||||||
return make_response(redirect(f"/user/{userdata[0]}"))
|
return make_response(redirect(f"/user/{userdata[0]}"))
|
||||||
page_userdata=db.get_user_info(page_userid)
|
res,page_userdata=db.get_user_info(page_userid)
|
||||||
|
if not res:
|
||||||
|
return errorpage(page_userdata) # page_userdata is error
|
||||||
|
|
||||||
# POST: Update display name or password
|
# POST: Update display name or password
|
||||||
if request.method == 'POST':
|
if request.method == 'POST':
|
||||||
|
@ -66,11 +72,13 @@ def userpage(page_userid:int=0):
|
||||||
return make_response(redirect('/'))
|
return make_response(redirect('/'))
|
||||||
|
|
||||||
# GET: return normal info page
|
# GET: return normal info page
|
||||||
return render_template("user.html", title="User Details",userdata=page_userdata,login_userid=userdata[0],userid=page_userid)
|
return render_template("user.html", title="User Details",login=logged_in,userdata=page_userdata,login_userid=userdata[0],userid=page_userid)
|
||||||
|
|
||||||
@app.route('/user/<uname>')
|
@app.route('/user/<uname>')
|
||||||
def user_redirect(uname:str):
|
def user_redirect(uname:str):
|
||||||
userdata=db.get_user_info_from_uname(uname)
|
res,userdata=db.get_user_info(uname)
|
||||||
|
if not res:
|
||||||
|
return errorpage(userdata) # userdata is error
|
||||||
if not userdata:
|
if not userdata:
|
||||||
return make_response(redirect(f"/user"))
|
return make_response(redirect(f"/user"))
|
||||||
return make_response(redirect(f"/user/{userdata[0]}"))
|
return make_response(redirect(f"/user/{userdata[0]}"))
|
||||||
|
@ -78,23 +86,21 @@ def user_redirect(uname:str):
|
||||||
@app.route('/add', methods=['GET','POST'])
|
@app.route('/add', methods=['GET','POST'])
|
||||||
def addpage():
|
def addpage():
|
||||||
# try to get userdata, else yeet to the homepage
|
# try to get userdata, else yeet to the homepage
|
||||||
logged_in,userdata=get_login_info(request.cookies.get('session'))
|
logged_in,userdata=get_login_info(request.cookies.get('session-id'))
|
||||||
if not logged_in:
|
if not logged_in:
|
||||||
return make_response(redirect('/'))
|
return make_response(redirect('/'))
|
||||||
# POST: check and add archive, show confirmation/error message at the end
|
# POST: check and add archive, show confirmation/error message at the end
|
||||||
if request.method == 'POST':
|
if request.method == 'POST':
|
||||||
postdict={}
|
postdict={}
|
||||||
# get and save all inputs, error if one doesn't exist or is wrong type
|
# get and save all inputs, error if one doesn't exist or is wrong type
|
||||||
for i,itype in [("name",str),("hash",str),("category",int),("size",float)]:
|
for i,itype in [("name",str),("hash",str),("category",int),("size",float),("size_multiplier",int)]:
|
||||||
try:
|
try:
|
||||||
postdict[i]=itype(request.form[i])
|
postdict[i]=itype(request.form[i])
|
||||||
|
if i == "size_multiplier":
|
||||||
|
postdict["size"]+=i
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
return errorpage("All fields need to be filled and don't play with their names!")
|
return errorpage("All fields need to be filled and don't play with their names!")
|
||||||
|
|
||||||
try:
|
|
||||||
postdict["size"]=postdict["size"]*int(request.form['size_multiplier'])
|
|
||||||
except Exception as e:
|
|
||||||
return errorpage("All fields need to be filled and don't play with their names!")
|
|
||||||
postdict["owner"]=userdata[0]
|
postdict["owner"]=userdata[0]
|
||||||
res,archid=db.add_archive(postdict)
|
res,archid=db.add_archive(postdict)
|
||||||
if res:
|
if res:
|
||||||
|
@ -111,16 +117,16 @@ def loginpage():
|
||||||
# POST: Process login request
|
# POST: Process login request
|
||||||
if request.method == 'POST':
|
if request.method == 'POST':
|
||||||
username=request.form['username']
|
username=request.form['username']
|
||||||
password=sha256(request.form['password'].encode()).hexdigest()
|
login_passhash=sha256(request.form['password'].encode()).hexdigest()
|
||||||
code,userid,passhash=db.get_passhash(username)
|
res,userid,db_passhash=db.get_passhash(username)
|
||||||
if code != 200:
|
if not res:
|
||||||
return passhash
|
return errorpage(userid)# userid is the error
|
||||||
# if passwords match, create session and return cookie
|
# if passwords match, create session and return cookie
|
||||||
if password.upper() == passhash.upper():
|
if login_passhash.upper() == db_passhash.upper():
|
||||||
lifetime=RAR_COOKIE_LIFETIME # lifetime of the sesskey in seconds
|
lifetime=RAR_COOKIE_LIFETIME # lifetime of the sesskey in seconds
|
||||||
sesskey=str(uuid())
|
sesskey=str(uuid())
|
||||||
db.set_sesskey(sesskey,userid,lifetime)
|
db.set_sesskey(sesskey,userid,lifetime)
|
||||||
resp=setcookie("session",sesskey,lifetime)
|
resp=setcookie("session-id",sesskey,lifetime)
|
||||||
return resp
|
return resp
|
||||||
else:
|
else:
|
||||||
return errorpage("You've entered the wrong password. This incident will be reported.")
|
return errorpage("You've entered the wrong password. This incident will be reported.")
|
||||||
|
@ -130,7 +136,7 @@ def loginpage():
|
||||||
|
|
||||||
@app.route('/logout')
|
@app.route('/logout')
|
||||||
def logout():
|
def logout():
|
||||||
sesskey=request.cookies.get('session')
|
sesskey=request.cookies.get('session-id')
|
||||||
logged_in,userdata=get_login_info(sesskey)
|
logged_in,userdata=get_login_info(sesskey)
|
||||||
if not logged_in:
|
if not logged_in:
|
||||||
return make_response(redirect('/login'))
|
return make_response(redirect('/login'))
|
||||||
|
@ -139,16 +145,21 @@ def logout():
|
||||||
|
|
||||||
@app.route('/view/<int:archid>')
|
@app.route('/view/<int:archid>')
|
||||||
def viewpage(archid:int):
|
def viewpage(archid:int):
|
||||||
logged_in,userdata=get_login_info(request.cookies.get('session'))
|
logged_in,userdata=get_login_info(request.cookies.get('session-id'))
|
||||||
archive,category,labels=db.get_archive_info(archid)
|
res,archive,category,labels=db.get_archive_info(archid)
|
||||||
|
if not res:
|
||||||
|
return errorpage(archive) # archive is error
|
||||||
return render_template("view.html", title="View Archive",userdata=userdata,login=logged_in,archive=archive,category=category,labels=labels)
|
return render_template("view.html", title="View Archive",userdata=userdata,login=logged_in,archive=archive,category=category,labels=labels)
|
||||||
|
|
||||||
@app.route('/delete/<int:archid>', methods=["GET","POST"])
|
@app.route('/delete/<int:archid>', methods=["GET","POST"])
|
||||||
def deletepage(archid:int):
|
def deletepage(archid:int):
|
||||||
logged_in,userdata=get_login_info(request.cookies.get('session'))
|
logged_in,userdata=get_login_info(request.cookies.get('session-id'))
|
||||||
archive,category,labels=db.get_archive_info(archid)
|
res,archive,category,labels=db.get_archive_info(archid)
|
||||||
|
if not res:
|
||||||
|
return errorpage(archive) # archive is error
|
||||||
if not logged_in or userdata[0] != archive[8]:
|
if not logged_in or userdata[0] != archive[8]:
|
||||||
return make_response(redirect(f"/view/{archid}"))
|
return make_response(redirect(f"/view/{archid}"))
|
||||||
|
# POST: check if input is correct and delete relevant data
|
||||||
if request.method == 'POST':
|
if request.method == 'POST':
|
||||||
if not request.form['archname'] == archive[1]:
|
if not request.form['archname'] == archive[1]:
|
||||||
return errorpage("The name input doesn't match!")
|
return errorpage("The name input doesn't match!")
|
||||||
|
@ -159,8 +170,10 @@ def deletepage(archid:int):
|
||||||
|
|
||||||
@app.route('/labels/<int:archid>', methods=["GET","POST"])
|
@app.route('/labels/<int:archid>', methods=["GET","POST"])
|
||||||
def labeleditpage(archid:int):
|
def labeleditpage(archid:int):
|
||||||
logged_in,userdata=get_login_info(request.cookies.get('session'))
|
logged_in,userdata=get_login_info(request.cookies.get('session-id'))
|
||||||
archive,category,labels=db.get_archive_info(archid)
|
res,archive,category,labels=db.get_archive_info(archid)
|
||||||
|
if not res:
|
||||||
|
return errorpage(archive) # archive is error
|
||||||
label_dict=db.get_label_labeltypes(category[3])
|
label_dict=db.get_label_labeltypes(category[3])
|
||||||
if not logged_in or userdata[0] != archive[8]:
|
if not logged_in or userdata[0] != archive[8]:
|
||||||
return make_response(redirect(f"/view/{archid}"))
|
return make_response(redirect(f"/view/{archid}"))
|
||||||
|
@ -168,7 +181,6 @@ def labeleditpage(archid:int):
|
||||||
# POST: parse everything and update labels
|
# POST: parse everything and update labels
|
||||||
if request.method == 'POST':
|
if request.method == 'POST':
|
||||||
on_labels=[]
|
on_labels=[]
|
||||||
print(request.form)
|
|
||||||
for i in request.form:
|
for i in request.form:
|
||||||
on_labels.append(i)
|
on_labels.append(i)
|
||||||
res, data=db.update_labels(archid, on_labels)
|
res, data=db.update_labels(archid, on_labels)
|
||||||
|
@ -185,7 +197,7 @@ def labeleditpage(archid:int):
|
||||||
@app.route('/search')
|
@app.route('/search')
|
||||||
def searchpage():
|
def searchpage():
|
||||||
# try to get userdata, else logout state
|
# try to get userdata, else logout state
|
||||||
logged_in,userdata=get_login_info(request.cookies.get('session'))
|
logged_in,userdata=get_login_info(request.cookies.get('session-id'))
|
||||||
|
|
||||||
# try to set all required variables, else defaults
|
# try to set all required variables, else defaults
|
||||||
try:
|
try:
|
||||||
|
@ -199,8 +211,7 @@ def searchpage():
|
||||||
category=0
|
category=0
|
||||||
label_dict={}
|
label_dict={}
|
||||||
try:
|
try:
|
||||||
keywords=request.args['q']
|
keywords="".join(request.args['q']).split(" ")
|
||||||
keywords="".join(keywords).split(" ")
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
keywords=[]
|
keywords=[]
|
||||||
try:
|
try:
|
||||||
|
@ -226,7 +237,7 @@ def searchpage():
|
||||||
########## FUNCTIONS
|
########## FUNCTIONS
|
||||||
|
|
||||||
def errorpage(message):
|
def errorpage(message):
|
||||||
return "<h2>ERROR: " + str(message) + "</h2>Go back and try again"
|
return "<h2>ERROR: " + str(message) + "</h2>Go back and try again", 400
|
||||||
|
|
||||||
## Checks if given sesskey is valid and returns user data
|
## Checks if given sesskey is valid and returns user data
|
||||||
## OUTPUT: (if sesskey valid) logged_in:bool=True, userdata:tuple
|
## OUTPUT: (if sesskey valid) logged_in:bool=True, userdata:tuple
|
||||||
|
@ -236,7 +247,9 @@ def get_login_info(sesskey:str):
|
||||||
return False,()
|
return False,()
|
||||||
logged_in,userid=db.check_sesskey(sesskey)
|
logged_in,userid=db.check_sesskey(sesskey)
|
||||||
if logged_in:
|
if logged_in:
|
||||||
userdata=db.get_user_info(userid)
|
res,userdata=db.get_user_info(userid)
|
||||||
|
if not res:
|
||||||
|
return errorpage(userdata) # userdata is error
|
||||||
else:
|
else:
|
||||||
userdata=()
|
userdata=()
|
||||||
return logged_in,userdata
|
return logged_in,userdata
|
||||||
|
@ -256,11 +269,11 @@ def get_category_selection(include_parents:bool=True):
|
||||||
if not cat[2]:
|
if not cat[2]:
|
||||||
if include_parents:
|
if include_parents:
|
||||||
htmlcatlist.append((cat[0],cat[1]))
|
htmlcatlist.append((cat[0],cat[1]))
|
||||||
parent=cat[1]
|
parentname=cat[1]
|
||||||
parentid=cat[0]
|
parentid=cat[0]
|
||||||
for i in catlist:
|
for i in catlist:
|
||||||
if i[2] == parentid:
|
if i[2] == parentid:
|
||||||
htmlcatlist.append((i[0],f"{parent}/{i[1]}"))
|
htmlcatlist.append((i[0],f"{parentname}/{i[1]}"))
|
||||||
return htmlcatlist
|
return htmlcatlist
|
||||||
|
|
||||||
## API CALLS (NO THANKS)
|
## API CALLS (NO THANKS)
|
||||||
|
|
111
flask/func.py
111
flask/func.py
|
@ -4,7 +4,7 @@ from os import environ
|
||||||
import time,re
|
import time,re
|
||||||
from config import *
|
from config import *
|
||||||
|
|
||||||
## params populated with environment variables, defaults can be changed for a permanent solution
|
## params populated with environment variables, get data from config if environment variables not set
|
||||||
conn_params={
|
conn_params={
|
||||||
"user" : environ.get('MARIADB_USER') if environ.get('MARIADB_USER') else MARIADB_USER,
|
"user" : environ.get('MARIADB_USER') if environ.get('MARIADB_USER') else MARIADB_USER,
|
||||||
"password" : environ.get('MARIADB_PASSWORD') if environ.get('MARIADB_PASSWORD') else MARIADB_PASSWORD,
|
"password" : environ.get('MARIADB_PASSWORD') if environ.get('MARIADB_PASSWORD') else MARIADB_PASSWORD,
|
||||||
|
@ -28,93 +28,91 @@ class db:
|
||||||
SIZE bigint NOT NULL,
|
SIZE bigint NOT NULL,
|
||||||
IMPORTED int,
|
IMPORTED int,
|
||||||
CATEGORY int,
|
CATEGORY int,
|
||||||
OWNER int
|
OWNER int);""")
|
||||||
);""")
|
|
||||||
self.cur.execute("""CREATE TABLE IF NOT EXISTS Users(
|
self.cur.execute("""CREATE TABLE IF NOT EXISTS Users(
|
||||||
ID int PRIMARY KEY AUTO_INCREMENT,
|
ID int PRIMARY KEY AUTO_INCREMENT,
|
||||||
UNAME text NOT NULL UNIQUE,
|
UNAME text NOT NULL UNIQUE,
|
||||||
DNAME text NOT NULL,
|
DNAME text NOT NULL,
|
||||||
CREATED int NOT NULL,
|
CREATED int NOT NULL,
|
||||||
STATE text,
|
STATE text,
|
||||||
PASSHASH text NOT NULL
|
PASSHASH text NOT NULL);""")
|
||||||
);""")
|
|
||||||
self.cur.execute("""CREATE TABLE IF NOT EXISTS Sessions(
|
self.cur.execute("""CREATE TABLE IF NOT EXISTS Sessions(
|
||||||
ID int PRIMARY KEY AUTO_INCREMENT,
|
ID int PRIMARY KEY AUTO_INCREMENT,
|
||||||
SESSKEY text NOT NULL UNIQUE,
|
SESSKEY text NOT NULL UNIQUE,
|
||||||
USERID int NOT NULL,
|
USERID int NOT NULL,
|
||||||
CREATED int NOT NULL,
|
CREATED int NOT NULL,
|
||||||
LIFE int
|
LIFE int);""")
|
||||||
);""")
|
|
||||||
self.cur.execute("""CREATE TABLE IF NOT EXISTS Cats(
|
self.cur.execute("""CREATE TABLE IF NOT EXISTS Cats(
|
||||||
ID int PRIMARY KEY AUTO_INCREMENT,
|
ID int PRIMARY KEY AUTO_INCREMENT,
|
||||||
CATEGORY text NOT NULL,
|
CATEGORY text NOT NULL,
|
||||||
PARENT int,
|
PARENT int,
|
||||||
DESCRIPTION text
|
DESCRIPTION text);""")
|
||||||
);""")
|
|
||||||
self.cur.execute("""CREATE TABLE IF NOT EXISTS ArchLab(
|
self.cur.execute("""CREATE TABLE IF NOT EXISTS ArchLab(
|
||||||
ID int PRIMARY KEY AUTO_INCREMENT,
|
ID int PRIMARY KEY AUTO_INCREMENT,
|
||||||
ARCHID int NOT NULL,
|
ARCHID int NOT NULL,
|
||||||
LABID int NOT NULL
|
LABID int NOT NULL);""")
|
||||||
);""")
|
|
||||||
self.cur.execute("""CREATE TABLE IF NOT EXISTS CatLabType(
|
self.cur.execute("""CREATE TABLE IF NOT EXISTS CatLabType(
|
||||||
ID int PRIMARY KEY AUTO_INCREMENT,
|
ID int PRIMARY KEY AUTO_INCREMENT,
|
||||||
CATID int NOT NULL,
|
CATID int NOT NULL,
|
||||||
LABID int NOT NULL
|
LABID int NOT NULL);""")
|
||||||
);""")
|
|
||||||
self.cur.execute("""CREATE TABLE IF NOT EXISTS Labs(
|
self.cur.execute("""CREATE TABLE IF NOT EXISTS Labs(
|
||||||
ID int PRIMARY KEY AUTO_INCREMENT,
|
ID int PRIMARY KEY AUTO_INCREMENT,
|
||||||
LABEL text NOT NULL,
|
LABEL text NOT NULL,
|
||||||
TYPE int NOT NULL
|
TYPE int NOT NULL);""")
|
||||||
);""")
|
|
||||||
self.cur.execute("""CREATE TABLE IF NOT EXISTS LabType(
|
self.cur.execute("""CREATE TABLE IF NOT EXISTS LabType(
|
||||||
ID int PRIMARY KEY AUTO_INCREMENT,
|
ID int PRIMARY KEY AUTO_INCREMENT,
|
||||||
NAME text NOT NULL,
|
NAME text NOT NULL,
|
||||||
DESCRIPTION text
|
DESCRIPTION text);""")
|
||||||
);""")
|
|
||||||
|
|
||||||
## Gets the passhash from a specific user
|
## Gets the passhash from a specific user
|
||||||
## OUTPUT: (If user exists) int=200, ID:int, passhash:str
|
## OUTPUT: (If user exists) bool=True, ID:int, passhash:str
|
||||||
## (If user does not exist) int=400, Exception:str
|
## (If user does not exist) bool=False, Exception:str
|
||||||
def get_passhash(self, username:str):
|
def get_passhash(self, username:str):
|
||||||
self.cur.execute(f"SELECT ID,PASSHASH FROM Users WHERE UNAME='{username}'")
|
self.cur.execute(f"SELECT ID,PASSHASH FROM Users WHERE UNAME='{username}'")
|
||||||
try:
|
resp=self.cur.fetchone()
|
||||||
resp=self.cur.fetchone()
|
if not resp:
|
||||||
except Exception as e:
|
return False, "The user does not exist!", None
|
||||||
return 400, e, NULL
|
return True, resp[0], resp[1]
|
||||||
return 200, resp[0], resp[1]
|
|
||||||
|
|
||||||
## Checks if sesskey exists and is not expired
|
## Checks if sesskey exists and is not expired
|
||||||
## OUTPUT: (if valiid) bool=True, USERID:str
|
## OUTPUT: (if valiid) bool=True, USERID:str
|
||||||
## (in invalid) bool=False, str=""
|
## (in invalid) bool=False, str=""
|
||||||
def check_sesskey(self, sesskey:str):
|
def check_sesskey(self, sesskey:str):
|
||||||
self.cur.execute(f"SELECT SESSKEY,USERID FROM Sessions WHERE SESSKEY='{sesskey}'")
|
self.cur.execute(f"SELECT SESSKEY,USERID,CREATED,LIFE FROM Sessions WHERE SESSKEY='{sesskey}'")
|
||||||
entry=self.cur.fetchone()
|
entry=self.cur.fetchone()
|
||||||
if entry and sesskey in entry:
|
if entry and sesskey in entry and time.time() < entry[2]+entry[3]:
|
||||||
return True, entry[1]
|
return True, entry[1]
|
||||||
else:
|
else:
|
||||||
return False, ""
|
return False, "The session key is expired or does not exist!"
|
||||||
|
|
||||||
## Sets a session key. That's it.
|
## Sets a session key. That's it.
|
||||||
def set_sesskey(self, sesskey:str, userid:int, lifetime:int):
|
def set_sesskey(self, sesskey:str, userid:int, lifetime:int):
|
||||||
self.cur.execute(f"INSERT INTO Sessions(SESSKEY,USERID,CREATED,LIFE) VALUES('{sesskey}',{userid},{time.time()},{lifetime})")
|
self.cur.execute(f"INSERT INTO Sessions(SESSKEY,USERID,CREATED,LIFE) VALUES('{sesskey}',{userid},{time.time()},{lifetime})")
|
||||||
|
|
||||||
|
## Removes the session key from the database (doesn't delete the cookie)
|
||||||
def logout_user(self, sesskey:str):
|
def logout_user(self, sesskey:str):
|
||||||
self.cur.execute(f"DELETE FROM Sessions WHERE SESSKEY='{sesskey}'")
|
self.cur.execute(f"DELETE FROM Sessions WHERE SESSKEY='{sesskey}'")
|
||||||
|
|
||||||
## Gets and returns all user info about one (1) user
|
## Gets and returns all user info about one (1) user
|
||||||
## OUTPUT: tuple=(ID:int,UNAME:str,DNAME:str,CREATED:int,STATE:text,PASSHASH:text)
|
## OUTPUT: (if successful) bool=True, tuple=(ID:int,UNAME:str,DNAME:str,CREATED:int,STATE:text,PASSHASH:text)
|
||||||
def get_user_info(self, userid:int):
|
## (if unsuccessful) bool=False, error:str
|
||||||
self.cur.execute(f"SELECT * FROM Users WHERE ID='{userid}'")
|
def get_user_info(self, identifier):
|
||||||
return self.cur.fetchone()
|
match identifier:
|
||||||
|
case int():
|
||||||
|
query=f"ID={identifier}"
|
||||||
|
case str():
|
||||||
|
query=f"UNAME='{identifier}'"
|
||||||
|
case _:
|
||||||
|
return False, "Wrong identifier type!"
|
||||||
|
|
||||||
## like above, just with uname
|
self.cur.execute(f"SELECT * FROM Users WHERE {query}")
|
||||||
## OUTPUT: tuple=(ID:int,UNAME:str,DNAME:str,CREATED:int,STATE:text,PASSHASH:text)
|
return True, self.cur.fetchone()
|
||||||
def get_user_info_from_uname(self, uname:str):
|
|
||||||
self.cur.execute(f"SELECT * FROM Users WHERE UNAME='{uname}'")
|
|
||||||
return self.cur.fetchone()
|
|
||||||
|
|
||||||
def update_user_info(self, userid, update_type:str,value):
|
## updates a value of a user in the database (if allowed in the dictionary)
|
||||||
allowed_types={"DNAME":str,"PASSHASH":str}
|
## OUTPUT: (if successful) bool=True, str="Updated"
|
||||||
|
## (if unsuccessful) bool=False, str="Not allowed"
|
||||||
|
def update_user_info(self, userid:int, update_type:str,value):
|
||||||
|
allowed_types={"DNAME":str,"PASSHASH":str} # only allow to edit these columns!
|
||||||
if update_type.upper() not in allowed_types:
|
if update_type.upper() not in allowed_types:
|
||||||
return False, "Not allowed"
|
return False, "Not allowed"
|
||||||
self.cur.execute(f"""UPDATE Users SET {update_type}={value if allowed_types[update_type]==int else f"'{value}'"} WHERE ID={userid}""")
|
self.cur.execute(f"""UPDATE Users SET {update_type}={value if allowed_types[update_type]==int else f"'{value}'"} WHERE ID={userid}""")
|
||||||
|
@ -141,13 +139,16 @@ class db:
|
||||||
archid=self.cur.fetchone()
|
archid=self.cur.fetchone()
|
||||||
return True,archid[0]
|
return True,archid[0]
|
||||||
|
|
||||||
|
## Deletes all relevant entries from Archs and ArchLab
|
||||||
def delete_archive(self, archid:int):
|
def delete_archive(self, archid:int):
|
||||||
self.cur.execute(f"""DELETE FROM Archs WHERE ID={archid}""")
|
self.cur.execute(f"""DELETE FROM Archs WHERE ID={archid}""")
|
||||||
self.cur.execute(f"""DELETE FROM ArchLab WHERE ARCHID={archid}""")
|
self.cur.execute(f"""DELETE FROM ArchLab WHERE ARCHID={archid}""")
|
||||||
|
|
||||||
## Returns all relevant information about one (1) archive
|
## Returns all relevant information about one (1) archive
|
||||||
## OUTPUT: archive:tuple=(ID:int,NAME:str,HASH:str,SIZE:int,IMPORTED[UNIX]:int,CATEGORY.ID:int,CATEGORY,str,CATEGORY.DESCRIPTION:str,USER.ID:int,DNAME:str),
|
## OUTPUT: (if archive doesn't exist) bool=False, error:str
|
||||||
## category:tuple=(ID:int,CATEGORY:str,DESCRIPTION:str,PID:int,PCAT:str,PDESC:str)
|
## (if archive exists) bool=True,
|
||||||
|
## archive:tuple=(ID:int,NAME:str,HASH:str,SIZE:int,IMPORTED[UNIX]:int,CATEGORY.ID:int,CATEGORY,str,CATEGORY.DESCRIPTION:str,USER.ID:int,DNAME:str),
|
||||||
|
## category:tuple=(ID:int,CATEGORY:str,DESCRIPTION:str,PID:int,PCAT:str,PDESC:str),
|
||||||
## labels:list=[…,(ID:int,LABEL:str,LABTYPE:int,LABDESC:str),…]
|
## labels:list=[…,(ID:int,LABEL:str,LABTYPE:int,LABDESC:str),…]
|
||||||
def get_archive_info(self, archid:int):
|
def get_archive_info(self, archid:int):
|
||||||
# get info about archive itself
|
# get info about archive itself
|
||||||
|
@ -156,6 +157,8 @@ class db:
|
||||||
JOIN Users ON Users.ID=Archs.OWNER
|
JOIN Users ON Users.ID=Archs.OWNER
|
||||||
WHERE Archs.ID='{archid}'""")
|
WHERE Archs.ID='{archid}'""")
|
||||||
archive=self.cur.fetchone()
|
archive=self.cur.fetchone()
|
||||||
|
if not archive:
|
||||||
|
return False, "The archive does to exist!", None, None
|
||||||
# get info about category and it's parent
|
# get info about category and it's parent
|
||||||
self.cur.execute(f"""SELECT c.ID,c.CATEGORY,c.DESCRIPTION,p.ID AS PID,p.CATEGORY as PCAT,p.DESCRIPTION AS PDESC FROM Cats c, Cats p
|
self.cur.execute(f"""SELECT c.ID,c.CATEGORY,c.DESCRIPTION,p.ID AS PID,p.CATEGORY as PCAT,p.DESCRIPTION AS PDESC FROM Cats c, Cats p
|
||||||
WHERE c.ID={archive[5]} AND c.PARENT=p.ID""")
|
WHERE c.ID={archive[5]} AND c.PARENT=p.ID""")
|
||||||
|
@ -163,8 +166,10 @@ class db:
|
||||||
# get info about labels of archive
|
# get info about labels of archive
|
||||||
|
|
||||||
labels=self.get_label_info(archid)
|
labels=self.get_label_info(archid)
|
||||||
return archive, category, labels
|
return True, archive, category, labels
|
||||||
|
|
||||||
|
## Gets all labels and their parents of an archive
|
||||||
|
## OUTPUT: list=[…,(ID:int,LABEL:str,LABTYPE:int,LABDESC:str),…]
|
||||||
def get_label_info(self, archid:int):
|
def get_label_info(self, archid:int):
|
||||||
self.cur.execute(f"""SELECT Labs.ID,Labs.LABEL,LabType.ID AS LABTYPE,LabType.DESCRIPTION AS LABDESC FROM ArchLab
|
self.cur.execute(f"""SELECT Labs.ID,Labs.LABEL,LabType.ID AS LABTYPE,LabType.DESCRIPTION AS LABDESC FROM ArchLab
|
||||||
JOIN Archs ON Archs.ID=ArchLab.ARCHID
|
JOIN Archs ON Archs.ID=ArchLab.ARCHID
|
||||||
|
@ -193,11 +198,11 @@ class db:
|
||||||
for w,e in labtypes_list:
|
for w,e in labtypes_list:
|
||||||
labtypes_ids.append(str(w))
|
labtypes_ids.append(str(w))
|
||||||
labtypes_names.append(e)
|
labtypes_names.append(e)
|
||||||
ltid_string="(" + ",".join(labtypes_ids) + ")"
|
ltids_string="(" + ",".join(labtypes_ids) + ")"
|
||||||
# gets all relevant labs: […,(ID:int,NAME:str,LTNAME:str),…]
|
# gets all relevant labs: […,(ID:int,NAME:str,LTNAME:str),…]
|
||||||
self.cur.execute(f"""SELECT Labs.ID,Labs.LABEL,LabType.NAME AS LTNAME FROM Labs
|
self.cur.execute(f"""SELECT Labs.ID,Labs.LABEL,LabType.NAME AS LTNAME FROM Labs
|
||||||
JOIN LabType ON Labs.TYPE=LabType.ID
|
JOIN LabType ON Labs.TYPE=LabType.ID
|
||||||
WHERE LabType.ID IN {ltid_string}
|
WHERE LabType.ID IN {ltids_string}
|
||||||
ORDER BY Labs.LABEL ASC""")
|
ORDER BY Labs.LABEL ASC""")
|
||||||
labs_list=self.cur.fetchall()
|
labs_list=self.cur.fetchall()
|
||||||
res_dict={}
|
res_dict={}
|
||||||
|
@ -211,7 +216,7 @@ class db:
|
||||||
|
|
||||||
## get a list of enabled labels and update the DB to reflect that state
|
## get a list of enabled labels and update the DB to reflect that state
|
||||||
## OUTPUT: (if on_labels empty) bool=False, str
|
## OUTPUT: (if on_labels empty) bool=False, str
|
||||||
## (else)
|
## (else) bool=True, str=""
|
||||||
def update_labels(self, archid:int, on_labels:list):
|
def update_labels(self, archid:int, on_labels:list):
|
||||||
# fail if no labels passed
|
# fail if no labels passed
|
||||||
if len(on_labels) == 0:
|
if len(on_labels) == 0:
|
||||||
|
@ -232,13 +237,14 @@ class db:
|
||||||
# remove all labels which are not on
|
# remove all labels which are not on
|
||||||
self.cur.execute(f"""DELETE FROM ArchLab WHERE ARCHID={archid} AND LABID NOT IN ({",".join(on_labels)})""")
|
self.cur.execute(f"""DELETE FROM ArchLab WHERE ARCHID={archid} AND LABID NOT IN ({",".join(on_labels)})""")
|
||||||
to_add_list=[]
|
to_add_list=[]
|
||||||
|
# creates all entries as strings and puts them into a list
|
||||||
for i in to_add:
|
for i in to_add:
|
||||||
to_add_list.append("(" + str(archid) + "," + str(i) + ")")
|
to_add_list.append("(" + str(archid) + "," + str(i) + ")")
|
||||||
# add all new labels
|
# add all new labels
|
||||||
self.cur.execute(f"""INSERT INTO ArchLab(ARCHID,LABID) VALUES{",".join(to_add_list)}""")
|
self.cur.execute(f"""INSERT INTO ArchLab(ARCHID,LABID) VALUES{",".join(to_add_list)}""")
|
||||||
return True, ""
|
return True, ""
|
||||||
|
|
||||||
## Returns n archives, sorted by (imported )time or size
|
## Returns n archives, sorted by a column
|
||||||
## OUTPUT: archives:array=[…,(ID:int,NAME:str,SIZE:str,IMPORTED[UNIX]:int),…]
|
## OUTPUT: archives:array=[…,(ID:int,NAME:str,SIZE:str,IMPORTED[UNIX]:int),…]
|
||||||
def get_n_archives(self, sorttype:str="time",category:int=0, keywords:list=[], count:int=20,labels:list=[]): # TODO: CLEANN!!!!!
|
def get_n_archives(self, sorttype:str="time",category:int=0, keywords:list=[], count:int=20,labels:list=[]): # TODO: CLEANN!!!!!
|
||||||
match sorttype:
|
match sorttype:
|
||||||
|
@ -271,21 +277,6 @@ class db:
|
||||||
{keyword_string}
|
{keyword_string}
|
||||||
ORDER BY {sorttype} LIMIT {count if count else 20}""")
|
ORDER BY {sorttype} LIMIT {count if count else 20}""")
|
||||||
archives=self.cur.fetchall()
|
archives=self.cur.fetchall()
|
||||||
## WARNING: JANK
|
|
||||||
#positive_archives=[]
|
|
||||||
#print("LABELS:", labels)
|
|
||||||
#if len(labels) >= 1:
|
|
||||||
# for arch in archives:
|
|
||||||
# archid=arch[0]
|
|
||||||
# archive_labels=self.get_label_info(archid)
|
|
||||||
# success=True
|
|
||||||
# for label in archive_labels:
|
|
||||||
# if not label[0] in labels:
|
|
||||||
# success=False
|
|
||||||
# if success:
|
|
||||||
# positive_archives.append(arch)
|
|
||||||
# if len(positive_archives) >= count:
|
|
||||||
# break
|
|
||||||
|
|
||||||
return archives
|
return archives
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue