2023-11-01 20:54:19 +00:00
|
|
|
"""Database."""
|
|
|
|
|
|
|
|
from datetime import datetime
|
2023-05-14 09:38:11 +01:00
|
|
|
|
|
|
|
import flask
|
|
|
|
import sqlalchemy
|
|
|
|
from sqlalchemy import create_engine, func
|
|
|
|
from sqlalchemy.engine import reflection
|
|
|
|
from sqlalchemy.orm import scoped_session, sessionmaker
|
|
|
|
|
|
|
|
session: sqlalchemy.orm.scoping.scoped_session = scoped_session(sessionmaker())
|
|
|
|
|
2023-11-01 20:54:19 +00:00
|
|
|
timeout = 2_000 # 20 seconds
|
2023-10-31 15:05:40 +00:00
|
|
|
|
2023-05-14 09:38:11 +01:00
|
|
|
|
|
|
|
def init_db(db_url: str, echo: bool = False) -> None:
|
|
|
|
"""Initialise database."""
|
|
|
|
session.configure(bind=get_engine(db_url, echo=echo))
|
|
|
|
|
|
|
|
|
|
|
|
def get_engine(db_url: str, echo: bool = False) -> sqlalchemy.engine.base.Engine:
|
2023-11-01 20:54:19 +00:00
|
|
|
"""Create an engine object."""
|
2023-10-31 15:05:40 +00:00
|
|
|
return create_engine(
|
|
|
|
db_url,
|
|
|
|
pool_recycle=3600,
|
|
|
|
echo=echo,
|
|
|
|
connect_args={
|
|
|
|
"options": f"-c lock_timeout={timeout} -c statement_timeout={timeout}"
|
|
|
|
},
|
|
|
|
)
|
2023-05-14 09:38:11 +01:00
|
|
|
|
|
|
|
|
|
|
|
def get_tables() -> list[str]:
|
|
|
|
"""Get a list of table names."""
|
|
|
|
tables: list[str] = reflection.Inspector.from_engine(session.bind).get_table_names()
|
|
|
|
return tables
|
|
|
|
|
|
|
|
|
|
|
|
def init_app(app: flask.app.Flask, echo: bool = False) -> None:
|
|
|
|
"""Initialise database connection within flask app."""
|
|
|
|
db_url = app.config["DB_URL"]
|
|
|
|
session.configure(bind=get_engine(db_url, echo=echo))
|
|
|
|
|
|
|
|
@app.teardown_appcontext
|
2023-11-01 20:54:19 +00:00
|
|
|
def shutdown_session(exception: BaseException | None = None) -> None:
|
2023-05-14 09:38:11 +01:00
|
|
|
session.remove()
|
|
|
|
|
|
|
|
|
2023-11-01 20:54:19 +00:00
|
|
|
def now_utc() -> sqlalchemy.sql.functions.Function[datetime]:
|
|
|
|
"""Now with UTC timezone."""
|
2023-05-14 09:38:11 +01:00
|
|
|
return func.timezone("utc", func.now())
|