2023-02-16 05:06:42 +00:00
|
|
|
from enum import Enum
|
2023-03-04 05:04:47 +00:00
|
|
|
from typing import Optional, List
|
|
|
|
from sqlalchemy import Table, Column
|
2023-02-16 05:06:42 +00:00
|
|
|
from sqlalchemy.types import Integer, String, TIMESTAMP
|
|
|
|
from sqlalchemy.sql import func
|
2023-02-19 20:40:25 +00:00
|
|
|
from sqlalchemy.dialects.mysql import insert
|
2023-03-04 05:04:47 +00:00
|
|
|
from sqlalchemy.sql import func, select
|
2023-02-19 20:40:25 +00:00
|
|
|
from sqlalchemy.engine import Row
|
2023-03-04 02:31:23 +00:00
|
|
|
import bcrypt
|
2023-02-16 05:06:42 +00:00
|
|
|
|
|
|
|
from core.data.schema.base import BaseData, metadata
|
|
|
|
|
|
|
|
aime_user = Table(
|
|
|
|
"aime_user",
|
|
|
|
metadata,
|
|
|
|
Column("id", Integer, nullable=False, primary_key=True, autoincrement=True),
|
|
|
|
Column("username", String(25), unique=True),
|
|
|
|
Column("email", String(255), unique=True),
|
|
|
|
Column("password", String(255)),
|
|
|
|
Column("permissions", Integer),
|
|
|
|
Column("created_date", TIMESTAMP, server_default=func.now()),
|
|
|
|
Column("last_login_date", TIMESTAMP, onupdate=func.now()),
|
|
|
|
Column("suspend_expire_time", TIMESTAMP),
|
|
|
|
mysql_charset='utf8mb4'
|
|
|
|
)
|
|
|
|
|
|
|
|
class PermissionBits(Enum):
|
|
|
|
PermUser = 1
|
|
|
|
PermMod = 2
|
|
|
|
PermSysAdmin = 4
|
|
|
|
|
|
|
|
class UserData(BaseData):
|
2023-02-19 20:40:25 +00:00
|
|
|
def create_user(self, id: int = None, username: str = None, email: str = None, password: str = None, permission: int = 1) -> Optional[int]:
|
|
|
|
if id is None:
|
|
|
|
sql = insert(aime_user).values(
|
|
|
|
username=username,
|
|
|
|
email=email,
|
|
|
|
password=password,
|
|
|
|
permissions=permission
|
|
|
|
)
|
2023-02-16 05:06:42 +00:00
|
|
|
else:
|
2023-02-19 20:40:25 +00:00
|
|
|
sql = insert(aime_user).values(
|
|
|
|
id=id,
|
|
|
|
username=username,
|
|
|
|
email=email,
|
|
|
|
password=password,
|
|
|
|
permissions=permission
|
|
|
|
)
|
2023-02-16 05:06:42 +00:00
|
|
|
|
2023-02-19 20:40:25 +00:00
|
|
|
conflict = sql.on_duplicate_key_update(
|
|
|
|
username=username,
|
|
|
|
email=email,
|
|
|
|
password=password,
|
|
|
|
permissions=permission
|
|
|
|
)
|
2023-02-16 05:06:42 +00:00
|
|
|
|
2023-02-19 20:40:25 +00:00
|
|
|
result = self.execute(conflict)
|
2023-02-16 05:06:42 +00:00
|
|
|
if result is None: return None
|
|
|
|
return result.lastrowid
|
|
|
|
|
2023-03-04 02:31:23 +00:00
|
|
|
def get_user(self, user_id: int) -> Optional[Row]:
|
|
|
|
sql = select(aime_user).where(aime_user.c.id == user_id)
|
2023-02-19 20:40:25 +00:00
|
|
|
result = self.execute(sql)
|
2023-03-04 02:31:23 +00:00
|
|
|
if result is None: return False
|
2023-02-19 20:40:25 +00:00
|
|
|
return result.fetchone()
|
|
|
|
|
2023-03-04 02:31:23 +00:00
|
|
|
def check_password(self, user_id: int, passwd: bytes = None) -> bool:
|
|
|
|
usr = self.get_user(user_id)
|
|
|
|
if usr is None: return False
|
2023-02-19 20:40:25 +00:00
|
|
|
|
2023-03-04 02:31:23 +00:00
|
|
|
if usr['password'] is None:
|
|
|
|
return False
|
2023-02-19 20:40:25 +00:00
|
|
|
|
2023-03-04 02:31:23 +00:00
|
|
|
return bcrypt.checkpw(passwd, usr['password'].encode())
|
2023-02-19 20:40:25 +00:00
|
|
|
|
2023-02-16 05:06:42 +00:00
|
|
|
def reset_autoincrement(self, ai_value: int) -> None:
|
2023-02-19 20:40:25 +00:00
|
|
|
# ALTER TABLE isn't in sqlalchemy so we do this the ugly way
|
2023-02-16 05:06:42 +00:00
|
|
|
sql = f"ALTER TABLE aime_user AUTO_INCREMENT={ai_value}"
|
2023-03-04 05:04:47 +00:00
|
|
|
self.execute(sql)
|
|
|
|
|
|
|
|
def delete_user(self, user_id: int) -> None:
|
|
|
|
sql = aime_user.delete(aime_user.c.id == user_id)
|
|
|
|
|
|
|
|
result = self.execute(sql)
|
|
|
|
if result is None:
|
|
|
|
self.logger.error(f"Failed to delete user with id {user_id}")
|
|
|
|
|
|
|
|
def get_unregistered_users(self) -> List[Row]:
|
|
|
|
"""
|
|
|
|
Returns a list of users who have not registered with the webui. They may or may not have cards.
|
|
|
|
"""
|
|
|
|
sql = select(aime_user).where(aime_user.c.password == None)
|
|
|
|
|
|
|
|
result = self.execute(sql)
|
|
|
|
if result is None:
|
|
|
|
return None
|
|
|
|
return result.fetchall()
|