parent
60192fda1d
commit
dda44b76e2
@ -1,4 +1,25 @@
|
||||
FROM ubuntu:latest
|
||||
LABEL authors="stranik"
|
||||
FROM python:3.10 AS compile-image
|
||||
|
||||
ENTRYPOINT ["top", "-b"]
|
||||
## install dependencies
|
||||
|
||||
RUN apt-get update && \
|
||||
apt-get install -y --no-install-recommends
|
||||
|
||||
## add and install requirements
|
||||
COPY requirements.txt .
|
||||
RUN pip install --upgrade pip && \
|
||||
pip install --no-cache-dir -r requirements.txt
|
||||
|
||||
|
||||
FROM python:3.10 AS build-image
|
||||
|
||||
# copy env from prev img
|
||||
COPY --from=compile-image /usr/local/lib/python3.10/site-packages /usr/local/lib/python3.10/site-packages
|
||||
|
||||
COPY . /app
|
||||
|
||||
EXPOSE 8000
|
||||
|
||||
WORKDIR /app
|
||||
|
||||
CMD ["sh", "-c" ,"python3 -m alembic upgrade head && python3 -m uvicorn main:app --host 0.0.0.0 --port 8000 --reload"]
|
||||
|
@ -0,0 +1,102 @@
|
||||
# A generic, single database configuration.
|
||||
|
||||
[alembic]
|
||||
# path to migration scripts
|
||||
script_location = db/migrations
|
||||
|
||||
# template used to generate migration files
|
||||
# file_template = %%(rev)s_%%(slug)s
|
||||
|
||||
# sys.path path, will be prepended to sys.path if present.
|
||||
# defaults to the current working directory.
|
||||
prepend_sys_path = .
|
||||
|
||||
# timezone to use when rendering the date within the migration file
|
||||
# as well as the filename.
|
||||
# If specified, requires the python-dateutil library that can be
|
||||
# installed by adding `alembic[tz]` to the pip requirements
|
||||
# string value is passed to dateutil.tz.gettz()
|
||||
# leave blank for localtime
|
||||
# timezone =
|
||||
|
||||
# max length of characters to apply to the
|
||||
# "slug" field
|
||||
# truncate_slug_length = 40
|
||||
|
||||
# set to 'true' to run the environment during
|
||||
# the 'revision' command, regardless of autogenerate
|
||||
# revision_environment = false
|
||||
|
||||
# set to 'true' to allow .pyc and .pyo files without
|
||||
# a source .py file to be detected as revisions in the
|
||||
# versions/ directory
|
||||
# sourceless = false
|
||||
|
||||
# version location specification; This defaults
|
||||
# to db/migrations/versions. When using multiple version
|
||||
# directories, initial revisions must be specified with --version-path.
|
||||
# The path separator used here should be the separator specified by "version_path_separator" below.
|
||||
# version_locations = %(here)s/bar:%(here)s/bat:db/migrations/versions
|
||||
|
||||
# version path separator; As mentioned above, this is the character used to split
|
||||
# version_locations. The default within new alembic.ini files is "os", which uses os.pathsep.
|
||||
# If this key is omitted entirely, it falls back to the legacy behavior of splitting on spaces and/or commas.
|
||||
# Valid values for version_path_separator are:
|
||||
#
|
||||
# version_path_separator = :
|
||||
# version_path_separator = ;
|
||||
# version_path_separator = space
|
||||
version_path_separator = os # Use os.pathsep. Default configuration used for new projects.
|
||||
|
||||
# the output encoding used when revision files
|
||||
# are written from script.py.mako
|
||||
# output_encoding = utf-8
|
||||
|
||||
sqlalchemy.url = driver://user:pass@localhost/dbname
|
||||
|
||||
|
||||
[post_write_hooks]
|
||||
# post_write_hooks defines scripts or Python functions that are run
|
||||
# on newly generated revision scripts. See the documentation for further
|
||||
# detail and examples
|
||||
|
||||
# format using "black" - use the console_scripts runner, against the "black" entrypoint
|
||||
# hooks = black
|
||||
# black.type = console_scripts
|
||||
# black.entrypoint = black
|
||||
# black.options = -l 79 REVISION_SCRIPT_FILENAME
|
||||
|
||||
# Logging configuration
|
||||
[loggers]
|
||||
keys = root,sqlalchemy,alembic
|
||||
|
||||
[handlers]
|
||||
keys = console
|
||||
|
||||
[formatters]
|
||||
keys = generic
|
||||
|
||||
[logger_root]
|
||||
level = WARN
|
||||
handlers = console
|
||||
qualname =
|
||||
|
||||
[logger_sqlalchemy]
|
||||
level = WARN
|
||||
handlers =
|
||||
qualname = sqlalchemy.engine
|
||||
|
||||
[logger_alembic]
|
||||
level = INFO
|
||||
handlers =
|
||||
qualname = alembic
|
||||
|
||||
[handler_console]
|
||||
class = StreamHandler
|
||||
args = (sys.stderr,)
|
||||
level = NOTSET
|
||||
formatter = generic
|
||||
|
||||
[formatter_generic]
|
||||
format = %(levelname)-5.5s [%(name)s] %(message)s
|
||||
datefmt = %H:%M:%S
|
@ -0,0 +1,11 @@
|
||||
from typing import Optional
|
||||
|
||||
from api.request.base import RequestBase
|
||||
from datetime import datetime
|
||||
|
||||
from pydantic import Field
|
||||
|
||||
|
||||
class AnalyticsFilters(RequestBase):
|
||||
start_time: Optional[datetime] = Field(None)
|
||||
end_time: Optional[datetime] = Field(None)
|
@ -0,0 +1,6 @@
|
||||
from pydantic import BaseModel
|
||||
|
||||
|
||||
class RequestBase(BaseModel):
|
||||
class Config:
|
||||
use_enum_values = True
|
@ -0,0 +1,18 @@
|
||||
from datetime import datetime
|
||||
|
||||
from api.response.base import ResponseBase
|
||||
|
||||
|
||||
class AreaCharts:
|
||||
name: str
|
||||
value: int
|
||||
|
||||
|
||||
class TimelineCharts(AreaCharts):
|
||||
time: datetime
|
||||
|
||||
|
||||
class RequestAnalytics(ResponseBase):
|
||||
timeline_charts: list[TimelineCharts]
|
||||
area_charts: list[AreaCharts]
|
||||
bar_chars: list[AreaCharts]
|
@ -0,0 +1,6 @@
|
||||
from pydantic import BaseModel
|
||||
|
||||
|
||||
class ResponseBase(BaseModel):
|
||||
class Config:
|
||||
use_enum_values = True
|
@ -0,0 +1,21 @@
|
||||
import os
|
||||
|
||||
from dotenv import load_dotenv
|
||||
|
||||
load_dotenv()
|
||||
|
||||
DB_HOST = os.environ.get("POSTGRES_HOST")
|
||||
DB_PORT = os.environ.get("POSTGRES_PORT")
|
||||
DB_NAME = os.environ.get("POSTGRES_DB")
|
||||
DB_USER = os.environ.get("POSTGRES_USER")
|
||||
DB_PASS = os.environ.get("POSTGRES_PASSWORD")
|
||||
|
||||
db_url = f"postgresql+asyncpg://{DB_USER}:{DB_PASS}@{DB_HOST}:{DB_PORT}/{DB_NAME}?async_fallback=True"
|
||||
|
||||
secret = os.environ.get("secret_key")
|
||||
encrypt_algorithm = os.environ.get('encrypt_algorithm')
|
||||
|
||||
SMTP_USER = os.environ.get("SMTP_USER")
|
||||
SMTP_PORT = os.environ.get("SMTP_PORT")
|
||||
SMTP_PASS = os.environ.get("SMTP_PASS")
|
||||
SMTP_SERVER = os.environ.get("SMTP_SERVER")
|
@ -0,0 +1 @@
|
||||
Generic single-database configuration.
|
@ -0,0 +1,85 @@
|
||||
import os
|
||||
from logging.config import fileConfig
|
||||
|
||||
from sqlalchemy import engine_from_config
|
||||
from sqlalchemy import pool
|
||||
|
||||
from alembic import context
|
||||
|
||||
from db.models.base import Base
|
||||
|
||||
from configs.config import db_url
|
||||
|
||||
from db.models.conveer import DBConveer
|
||||
from db.models.camera import DBCamera
|
||||
|
||||
# this is the Alembic Config object, which provides
|
||||
# access to the values within the .ini file in use.
|
||||
config = context.config
|
||||
|
||||
# Interpret the config file for Python logging.
|
||||
# This line sets up loggers basically.
|
||||
fileConfig(config.config_file_name) # type: ignore
|
||||
|
||||
# add your model's MetaData object here
|
||||
# for 'autogenerate' support
|
||||
target_metadata = Base.metadata
|
||||
|
||||
config.set_main_option("sqlalchemy.url", db_url)
|
||||
|
||||
# other values from the config, defined by the needs of env.py,
|
||||
# can be acquired:
|
||||
# my_important_option = config.get_main_option("my_important_option")
|
||||
# ... etc.
|
||||
|
||||
|
||||
def run_migrations_offline() -> None:
|
||||
"""Run migrations in 'offline' mode.
|
||||
|
||||
This configures the context with just a URL
|
||||
and not an Engine, though an Engine is acceptable
|
||||
here as well. By skipping the Engine creation
|
||||
we don't even need a DBAPI to be available.
|
||||
|
||||
Calls to context.execute() here emit the given string to the
|
||||
script output.
|
||||
|
||||
"""
|
||||
url = config.get_main_option("sqlalchemy.url")
|
||||
context.configure(
|
||||
url=url,
|
||||
target_metadata=target_metadata,
|
||||
literal_binds=True,
|
||||
dialect_opts={"paramstyle": "named"},
|
||||
)
|
||||
|
||||
with context.begin_transaction():
|
||||
context.run_migrations()
|
||||
|
||||
|
||||
def run_migrations_online() -> None:
|
||||
"""Run migrations in 'online' mode.
|
||||
|
||||
In this scenario we need to create an Engine
|
||||
and associate a connection with the context.
|
||||
|
||||
"""
|
||||
connectable = engine_from_config(
|
||||
config.get_section(config.config_ini_section, {}),
|
||||
prefix="sqlalchemy.",
|
||||
poolclass=pool.NullPool,
|
||||
)
|
||||
|
||||
with connectable.connect() as connection:
|
||||
context.configure(
|
||||
connection=connection, target_metadata=target_metadata
|
||||
)
|
||||
|
||||
with context.begin_transaction():
|
||||
context.run_migrations()
|
||||
|
||||
|
||||
if context.is_offline_mode():
|
||||
run_migrations_offline()
|
||||
else:
|
||||
run_migrations_online()
|
@ -0,0 +1,24 @@
|
||||
"""${message}
|
||||
|
||||
Revision ID: ${up_revision}
|
||||
Revises: ${down_revision | comma,n}
|
||||
Create Date: ${create_date}
|
||||
|
||||
"""
|
||||
from alembic import op
|
||||
import sqlalchemy as sa
|
||||
${imports if imports else ""}
|
||||
|
||||
# revision identifiers, used by Alembic.
|
||||
revision = ${repr(up_revision)}
|
||||
down_revision = ${repr(down_revision)}
|
||||
branch_labels = ${repr(branch_labels)}
|
||||
depends_on = ${repr(depends_on)}
|
||||
|
||||
|
||||
def upgrade():
|
||||
${upgrades if upgrades else "pass"}
|
||||
|
||||
|
||||
def downgrade():
|
||||
${downgrades if downgrades else "pass"}
|
@ -0,0 +1,52 @@
|
||||
"""initial
|
||||
|
||||
Revision ID: 3c651a0d1fe0
|
||||
Revises:
|
||||
Create Date: 2023-10-27 18:19:47.402501
|
||||
|
||||
"""
|
||||
from alembic import op
|
||||
import sqlalchemy as sa
|
||||
|
||||
|
||||
# revision identifiers, used by Alembic.
|
||||
revision = '3c651a0d1fe0'
|
||||
down_revision = None
|
||||
branch_labels = None
|
||||
depends_on = None
|
||||
|
||||
|
||||
def upgrade():
|
||||
# ### commands auto generated by Alembic - please adjust! ###
|
||||
op.create_table('camera',
|
||||
sa.Column('camera_type', sa.String(), nullable=False),
|
||||
sa.Column('order_numb', sa.Integer(), nullable=True),
|
||||
sa.Column('id', sa.Integer(), autoincrement=True, nullable=False),
|
||||
sa.Column('created_at', sa.TIMESTAMP(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False),
|
||||
sa.Column('updated_at', sa.TIMESTAMP(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False),
|
||||
sa.PrimaryKeyConstraint('id', name=op.f('pk_camera')),
|
||||
sa.UniqueConstraint('id', name=op.f('uq_camera_id')),
|
||||
sa.UniqueConstraint('order_numb', name=op.f('uq_camera_order_numb'))
|
||||
)
|
||||
op.create_table('conveer',
|
||||
sa.Column('wood', sa.Integer(), nullable=True),
|
||||
sa.Column('metal', sa.Integer(), nullable=True),
|
||||
sa.Column('glass', sa.Integer(), nullable=True),
|
||||
sa.Column('plastic', sa.Integer(), nullable=True),
|
||||
sa.Column('camera_id', sa.Integer(), nullable=False),
|
||||
sa.Column('id', sa.Integer(), autoincrement=True, nullable=False),
|
||||
sa.Column('created_at', sa.TIMESTAMP(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False),
|
||||
sa.Column('updated_at', sa.TIMESTAMP(), server_default=sa.text('CURRENT_TIMESTAMP'), nullable=False),
|
||||
sa.ForeignKeyConstraint(['camera_id'], ['camera.id'], name=op.f('fk_conveer_camera_id_camera')),
|
||||
sa.PrimaryKeyConstraint('id', name=op.f('pk_conveer')),
|
||||
sa.UniqueConstraint('id', name=op.f('uq_conveer_id'))
|
||||
)
|
||||
op.execute('''INSERT INTO camera(id,order_numb, camera_type) VALUES (1,1, 'По умолчанию')''')
|
||||
# ### end Alembic commands ###
|
||||
|
||||
|
||||
def downgrade():
|
||||
# ### commands auto generated by Alembic - please adjust! ###
|
||||
op.drop_table('conveer')
|
||||
op.drop_table('camera')
|
||||
# ### end Alembic commands ###
|
@ -0,0 +1,44 @@
|
||||
import datetime
|
||||
from typing import Optional
|
||||
|
||||
import sqlalchemy.types as types
|
||||
from sqlalchemy import Column, text, MetaData
|
||||
from sqlalchemy.orm import declarative_base
|
||||
|
||||
meta = MetaData(naming_convention={
|
||||
"ix": "ix_%(column_0_label)s",
|
||||
"uq": "uq_%(table_name)s_%(column_0_name)s",
|
||||
"ck": "ck_%(table_name)s_%(constraint_name)s",
|
||||
"fk": "fk_%(table_name)s_%(column_0_name)s_%(referred_table_name)s",
|
||||
"pk": "pk_%(table_name)s"
|
||||
}, schema=None)
|
||||
|
||||
Base = declarative_base(metadata=meta) # type: ignore
|
||||
|
||||
|
||||
class BaseModel(Base):
|
||||
__abstract__ = True
|
||||
|
||||
id = Column(types.Integer, nullable=False, unique=True, primary_key=True, autoincrement=True)
|
||||
created_at = Column(
|
||||
types.TIMESTAMP, nullable=False, server_default=text('CURRENT_TIMESTAMP')
|
||||
)
|
||||
updated_at = Column(
|
||||
types.TIMESTAMP, nullable=False, server_default=text('CURRENT_TIMESTAMP'), onupdate=text('CURRENT_TIMESTAMP')
|
||||
)
|
||||
|
||||
__mapper_args__ = {"eager_defaults": True}
|
||||
|
||||
@property
|
||||
def created_at_timestamp(self) -> int:
|
||||
return int(self.created_at.timestamp())
|
||||
|
||||
@property
|
||||
def updated_at_timestamp(self) -> int:
|
||||
return int(self.updated_at.timestamp())
|
||||
|
||||
def set_updated_at(self, date_time: Optional[datetime.datetime] = None) -> None:
|
||||
if date_time:
|
||||
self.updated_at = date_time
|
||||
else:
|
||||
self.updated_at = datetime.datetime.now()
|
@ -0,0 +1,12 @@
|
||||
from db.models.base import BaseModel
|
||||
from sqlalchemy import (
|
||||
Column,
|
||||
Integer,
|
||||
String
|
||||
)
|
||||
|
||||
|
||||
class DBCamera(BaseModel):
|
||||
__tablename__ = "camera"
|
||||
camera_type = Column(String, nullable=False)
|
||||
order_numb = Column(Integer, unique=True)
|
@ -0,0 +1,22 @@
|
||||
from db.models.base import BaseModel
|
||||
from sqlalchemy import (
|
||||
Column,
|
||||
Integer,
|
||||
ForeignKey
|
||||
)
|
||||
|
||||
from sqlalchemy.orm import relationship
|
||||
from db.models.camera import DBCamera
|
||||
|
||||
|
||||
class DBConveer(BaseModel):
|
||||
__tablename__ = "conveer"
|
||||
wood = Column(Integer)
|
||||
metal = Column(Integer)
|
||||
glass = Column(Integer)
|
||||
plastic = Column(Integer)
|
||||
camera_id = Column(Integer, ForeignKey('camera.id'), nullable=False)
|
||||
|
||||
camera = relationship('DBCamera', lazy="raise", uselist=False)
|
||||
|
||||
|
@ -0,0 +1,31 @@
|
||||
from datetime import datetime
|
||||
from typing import Optional
|
||||
|
||||
from db.repository.base import BaseRepository
|
||||
|
||||
from db.models.conveer import DBConveer
|
||||
|
||||
from sqlalchemy import (
|
||||
select,
|
||||
func
|
||||
)
|
||||
|
||||
|
||||
class AnalyticsRepository(BaseRepository):
|
||||
|
||||
async def get_analytic(self,
|
||||
start_date: Optional[datetime],
|
||||
end_date: Optional[datetime]):
|
||||
query = (
|
||||
select(DBConveer)
|
||||
.select_from(DBConveer)
|
||||
)
|
||||
|
||||
if start_date is not None:
|
||||
query = query.filter(DBConveer.created_at >= func.timezone('UTC', start_date))
|
||||
|
||||
if end_date is not None:
|
||||
query = query.filter(DBConveer.created_at <= func.timezone('UTC', end_date))
|
||||
|
||||
|
||||
return await self.all_ones(query)
|
@ -0,0 +1,76 @@
|
||||
from typing import Any, Union
|
||||
|
||||
from sqlalchemy import select, exists
|
||||
from sqlalchemy.engine import ChunkedIteratorResult
|
||||
from sqlalchemy.exc import IntegrityError
|
||||
from sqlalchemy.ext.asyncio import AsyncSession
|
||||
from sqlalchemy.sql import Select, Delete
|
||||
|
||||
from db.models.base import BaseModel
|
||||
|
||||
|
||||
class BaseRepository:
|
||||
def __init__(self, session: AsyncSession) -> None:
|
||||
self._session = session
|
||||
|
||||
async def execute(self, query: Select) -> ChunkedIteratorResult:
|
||||
return await self._session.execute(query)
|
||||
|
||||
async def execute_parametrize(self, query: Union[str, Select], params: dict) -> ChunkedIteratorResult:
|
||||
return await self._session.execute(statement=query, params=params)
|
||||
|
||||
async def one(self, query: Select) -> Any:
|
||||
result = await self.execute(query)
|
||||
return result.one()
|
||||
|
||||
async def one_or_none(self, query: Select) -> Any:
|
||||
result = await self.execute(query)
|
||||
return result.one_or_none()
|
||||
|
||||
async def one_val(self, query: Select) -> Any:
|
||||
result = await self.one(query)
|
||||
return result[0]
|
||||
|
||||
async def one_or_none_val(self, query: Select) -> Any:
|
||||
result = await self.one_or_none(query)
|
||||
if not result:
|
||||
return None
|
||||
return result[0]
|
||||
|
||||
async def add_model(self, model: BaseModel) -> None:
|
||||
self._session.add(model)
|
||||
await self._session.commit()
|
||||
|
||||
async def refresh_model(self, model: BaseModel):
|
||||
await self._session.refresh(model)
|
||||
|
||||
async def add_model_ignore_exceptions(self, model: BaseModel) -> None:
|
||||
try:
|
||||
async with self._session.begin_nested():
|
||||
self._session.add(model)
|
||||
except IntegrityError:
|
||||
pass
|
||||
|
||||
async def add_models(self, models: list[BaseModel]) -> None:
|
||||
for model in models:
|
||||
await self.add_model(model)
|
||||
await self._session.commit()
|
||||
|
||||
async def delete(self, model: BaseModel) -> None:
|
||||
await self._session.delete(model)
|
||||
|
||||
async def delete_many(self, models: list[BaseModel]) -> None:
|
||||
for model in models:
|
||||
await self.delete(model)
|
||||
|
||||
async def all(self, query: Select) -> list[Any]:
|
||||
result = await self.execute(query)
|
||||
return result.all()
|
||||
|
||||
async def all_ones(self, query: Select) -> list[Any]:
|
||||
result = await self.execute(query)
|
||||
return [row[0] for row in result.all()]
|
||||
|
||||
async def exists(self, query: Select) -> bool:
|
||||
query = select(exists(query))
|
||||
return await self.one_val(query)
|
@ -0,0 +1,14 @@
|
||||
import uvicorn
|
||||
from fastapi import FastAPI
|
||||
|
||||
from server.routers_init import all_routers
|
||||
|
||||
app = FastAPI(
|
||||
title="Конвеер"
|
||||
)
|
||||
|
||||
for router in all_routers:
|
||||
app.include_router(router)
|
||||
|
||||
if __name__ == "__main__":
|
||||
uvicorn.run(app="main:app", reload=True)
|
@ -0,0 +1,18 @@
|
||||
from datetime import datetime
|
||||
from typing import Optional
|
||||
|
||||
from sqlalchemy.ext.asyncio import AsyncSession
|
||||
|
||||
from db.repository.analytics import AnalyticsRepository
|
||||
|
||||
|
||||
class AnalyticsManager:
|
||||
|
||||
@classmethod
|
||||
async def get_by_filters(cls,
|
||||
session: AsyncSession,
|
||||
start_date: Optional[datetime],
|
||||
end_date: Optional[datetime]):
|
||||
datas = await AnalyticsRepository(session).get_analytic(start_date=start_date,
|
||||
end_date=end_date)
|
||||
return datas
|
@ -0,0 +1,10 @@
|
||||
fastapi
|
||||
uvicorn
|
||||
SQLAlchemy
|
||||
psycopg2-binary
|
||||
asyncpg
|
||||
alembic
|
||||
python-dotenv
|
||||
python-jose
|
||||
python-multipart
|
||||
pytz
|
@ -0,0 +1,26 @@
|
||||
from datetime import datetime, timedelta
|
||||
from typing import Optional
|
||||
|
||||
from sqlalchemy.ext.asyncio import AsyncSession
|
||||
|
||||
from fastapi import Depends, Query
|
||||
|
||||
from configs.config import secret, encrypt_algorithm
|
||||
|
||||
from vendors.db import async_session
|
||||
|
||||
from fastapi.exceptions import HTTPException
|
||||
|
||||
|
||||
async def get_session() -> AsyncSession:
|
||||
async with async_session() as session:
|
||||
yield session
|
||||
|
||||
class PagesPaginationParams:
|
||||
def __init__(
|
||||
self,
|
||||
limit: int = Query(50, ge=0, le=1_000),
|
||||
offset: int = Query(0, ge=0, alias='skip'),
|
||||
) -> None:
|
||||
self.limit = limit
|
||||
self.offset = offset
|
@ -0,0 +1,22 @@
|
||||
from fastapi import APIRouter, Depends
|
||||
from sqlalchemy.ext.asyncio import AsyncSession
|
||||
|
||||
from api.request.analytics import AnalyticsFilters
|
||||
from managers.analytics import AnalyticsManager
|
||||
from server.depends import get_session, PagesPaginationParams
|
||||
|
||||
router = APIRouter(prefix="/analytics", tags=['Ride'])
|
||||
|
||||
|
||||
@router.post('/all')
|
||||
async def get_all_analytics(
|
||||
filters: AnalyticsFilters,
|
||||
session: AsyncSession = Depends(get_session),
|
||||
):
|
||||
data = await AnalyticsManager.get_by_filters(
|
||||
session=session,
|
||||
start_date=filters.start_time,
|
||||
end_date=filters.end_time
|
||||
)
|
||||
|
||||
return data
|
@ -0,0 +1,3 @@
|
||||
from server.routers.analytics import router as analytics_router
|
||||
|
||||
all_routers = [analytics_router]
|
@ -0,0 +1,22 @@
|
||||
from typing import AsyncGenerator
|
||||
|
||||
from sqlalchemy.ext.asyncio import AsyncSession, create_async_engine
|
||||
from sqlalchemy.ext.declarative import declarative_base
|
||||
from sqlalchemy.orm import sessionmaker
|
||||
|
||||
from configs.config import db_url
|
||||
|
||||
DATABASE_URL = db_url
|
||||
Base = declarative_base()
|
||||
|
||||
engine = create_async_engine(DATABASE_URL)
|
||||
async_session = sessionmaker(
|
||||
bind=engine,
|
||||
class_=AsyncSession,
|
||||
expire_on_commit=False,
|
||||
)
|
||||
|
||||
|
||||
async def get_async_session() -> AsyncGenerator[AsyncSession, None]:
|
||||
async with async_session() as session:
|
||||
yield session
|
Loading…
Reference in new issue