initial commit
This commit is contained in:
1
api/README.md
Normal file
1
api/README.md
Normal file
@@ -0,0 +1 @@
|
||||
#connect/api
|
0
api/api/__init__.py
Normal file
0
api/api/__init__.py
Normal file
80
api/api/__main__.py
Normal file
80
api/api/__main__.py
Normal file
@@ -0,0 +1,80 @@
|
||||
import sys
|
||||
import logging
|
||||
|
||||
import loguru
|
||||
from fastapi import FastAPI
|
||||
from fastapi.middleware.cors import CORSMiddleware
|
||||
from uvicorn import run
|
||||
|
||||
from api.config import get_settings, DefaultSettings
|
||||
from api.endpoints import list_of_routes
|
||||
from api.utils.common import get_hostname
|
||||
|
||||
logger = logging.getLogger()
|
||||
logger.setLevel(logging.INFO)
|
||||
|
||||
|
||||
def bind_routes(application: FastAPI, setting: DefaultSettings) -> None:
|
||||
"""Bind all routes to application."""
|
||||
for route in list_of_routes:
|
||||
application.include_router(route, prefix=setting.PATH_PREFIX)
|
||||
|
||||
|
||||
def get_app() -> FastAPI:
|
||||
"""Creates application and all dependable objects."""
|
||||
loguru.logger.remove()
|
||||
loguru.logger.add(sys.stderr, level="WARNING")
|
||||
loguru.logger.add(sys.stdout, level="DEBUG")
|
||||
description = "Эндпоинты"
|
||||
|
||||
application = FastAPI(
|
||||
title="Connect API",
|
||||
description=description,
|
||||
docs_url="/swagger",
|
||||
openapi_url="/openapi",
|
||||
version="0.1.0",
|
||||
)
|
||||
settings = get_settings()
|
||||
bind_routes(application, settings)
|
||||
application.state.settings = settings
|
||||
return application
|
||||
|
||||
|
||||
app = get_app()
|
||||
|
||||
dev_origins = [
|
||||
"http://localhost:3000",
|
||||
]
|
||||
|
||||
prod_origins = [""]
|
||||
|
||||
origins = dev_origins if get_settings().ENV == "local" else prod_origins
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
settings_for_application = get_settings()
|
||||
if settings_for_application.ENV == "prod":
|
||||
run(
|
||||
"api.__main__:app",
|
||||
host=get_hostname(settings_for_application.APP_HOST),
|
||||
port=settings_for_application.APP_PORT,
|
||||
reload=False,
|
||||
log_level="info",
|
||||
)
|
||||
else:
|
||||
run(
|
||||
"api.__main__:app",
|
||||
host=get_hostname(settings_for_application.APP_HOST),
|
||||
port=settings_for_application.APP_PORT,
|
||||
reload=True,
|
||||
reload_dirs=["api", "tests"],
|
||||
log_level="info",
|
||||
)
|
||||
|
||||
app.add_middleware(
|
||||
CORSMiddleware,
|
||||
allow_origins=origins,
|
||||
allow_credentials=True,
|
||||
allow_methods=["GET", "POST", "OPTIONS", "DELETE", "PUT"],
|
||||
allow_headers=["*"],
|
||||
)
|
7
api/api/config/__init__.py
Normal file
7
api/api/config/__init__.py
Normal file
@@ -0,0 +1,7 @@
|
||||
from .default import DefaultSettings
|
||||
from .utils import get_settings
|
||||
|
||||
__all__ = [
|
||||
"DefaultSettings",
|
||||
"get_settings",
|
||||
]
|
43
api/api/config/default.py
Normal file
43
api/api/config/default.py
Normal file
@@ -0,0 +1,43 @@
|
||||
import uuid
|
||||
|
||||
from os import environ
|
||||
|
||||
from pydantic import BaseModel
|
||||
from pydantic_settings import BaseSettings
|
||||
|
||||
|
||||
class DbCredentialsSchema(BaseModel):
|
||||
host: str
|
||||
user: str
|
||||
password: str
|
||||
database: str
|
||||
port: int
|
||||
|
||||
|
||||
class DefaultSettings(BaseSettings):
|
||||
ENV: str = environ.get("ENV", "local")
|
||||
PATH_PREFIX: str = environ.get("PATH_PREFIX", "/api/v1")
|
||||
APP_HOST: str = environ.get("APP_HOST", "http://127.0.0.1")
|
||||
APP_PORT: int = int(environ.get("APP_PORT", 8000))
|
||||
APP_ID: uuid.UUID = environ.get("APP_ID", uuid.uuid4())
|
||||
LOGS_STORAGE_PATH: str = environ.get("LOGS_STORAGE_PATH", "storage/logs")
|
||||
|
||||
MYSQL_DB: str = environ.get("MYSQL_DB", "connect_test")
|
||||
MYSQL_HOST: str = environ.get("MYSQL_HOST", "localhost")
|
||||
MYSQL_USER: str = environ.get("MYSQL_USER", "connect")
|
||||
MYSQL_PORT: int = int(environ.get("MYSQL_PORT", "3306")[-4:])
|
||||
MYSQL_PASSWORD: str = environ.get("MYSQL_PASSWORD", "hackme")
|
||||
CONNECTION_POOL_SIZE: int = int(environ.get("CONNECTION_POOL_SIZE", "30"))
|
||||
CONNECTION_OVERFLOW: int = int(environ.get("CONNECTION_OVERFLOW", "100"))
|
||||
|
||||
DOMAIN: str = environ.get("DOMAIN", "localhost")
|
||||
BROKER_PROTOCOL: str = environ.get("BROKER_PROTOCOL", "amqp")
|
||||
BROKER_HOST: str = environ.get("BROKER_HOST", "localhost")
|
||||
BROKER_USER: str = environ.get("BROKER_USER", "guest")
|
||||
BROKER_PORT: int = int(environ.get("BROKER_PORT", "5672"))
|
||||
BROKER_PASSWORD: str = environ.get("BROKER_PASSWORD", "guest")
|
||||
BROKER_RABBITMQ_VHOST: str = environ.get("BROKER_RABBITMQ_VHOST", "")
|
||||
|
||||
class Config:
|
||||
# env_file = "../.env"
|
||||
env_file_encoding = "utf-8"
|
15
api/api/config/utils.py
Normal file
15
api/api/config/utils.py
Normal file
@@ -0,0 +1,15 @@
|
||||
from os import environ
|
||||
|
||||
from api.config.default import DefaultSettings
|
||||
|
||||
|
||||
def get_settings() -> DefaultSettings:
|
||||
env = environ.get("ENV", "local")
|
||||
env_settings = {
|
||||
"local": DefaultSettings,
|
||||
"prod": None,
|
||||
}
|
||||
try:
|
||||
return env_settings[env]()
|
||||
except KeyError:
|
||||
return DefaultSettings()
|
7
api/api/db/__init__.py
Normal file
7
api/api/db/__init__.py
Normal file
@@ -0,0 +1,7 @@
|
||||
from sqlalchemy import MetaData
|
||||
|
||||
metadata = MetaData()
|
||||
|
||||
__all__ = [
|
||||
"metadata",
|
||||
]
|
119
api/api/db/alembic.ini
Normal file
119
api/api/db/alembic.ini
Normal file
@@ -0,0 +1,119 @@
|
||||
# A generic, single database configuration.
|
||||
|
||||
[alembic]
|
||||
# path to migration scripts
|
||||
# Use forward slashes (/) also on windows to provide an os agnostic path
|
||||
script_location = alembic
|
||||
|
||||
# template used to generate migration file names; The default value is %%(rev)s_%%(slug)s
|
||||
# Uncomment the line below if you want the files to be prepended with date and time
|
||||
# see https://alembic.sqlalchemy.org/en/latest/tutorial.html#editing-the-ini-file
|
||||
# for all available tokens
|
||||
# file_template = %%(year)d_%%(month).2d_%%(day).2d_%%(hour).2d%%(minute).2d-%%(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>=3.9 or backports.zoneinfo library and tzdata library.
|
||||
# Any required deps can installed by adding `alembic[tz]` to the pip requirements
|
||||
# string value is passed to ZoneInfo()
|
||||
# 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 alembic/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:alembic/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 = newline
|
||||
#
|
||||
# Use os.pathsep. Default configuration used for new projects.
|
||||
version_path_separator = os
|
||||
|
||||
# set to 'true' to search source files recursively
|
||||
# in each "version_locations" directory
|
||||
# new in Alembic version 1.10
|
||||
# recursive_version_locations = false
|
||||
|
||||
# the output encoding used when revision files
|
||||
# are written from script.py.mako
|
||||
# output_encoding = utf-8
|
||||
|
||||
sqlalchemy.url = sqlite:///db.sqlite3
|
||||
|
||||
|
||||
[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
|
||||
|
||||
# lint with attempts to fix using "ruff" - use the exec runner, execute a binary
|
||||
# hooks = ruff
|
||||
# ruff.type = exec
|
||||
# ruff.executable = %(here)s/.venv/bin/ruff
|
||||
# ruff.options = check --fix REVISION_SCRIPT_FILENAME
|
||||
|
||||
# Logging configuration
|
||||
[loggers]
|
||||
keys = root,sqlalchemy,alembic
|
||||
|
||||
[handlers]
|
||||
keys = console
|
||||
|
||||
[formatters]
|
||||
keys = generic
|
||||
|
||||
[logger_root]
|
||||
level = WARNING
|
||||
handlers = console
|
||||
qualname =
|
||||
|
||||
[logger_sqlalchemy]
|
||||
level = WARNING
|
||||
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
|
1
api/api/db/alembic/README
Normal file
1
api/api/db/alembic/README
Normal file
@@ -0,0 +1 @@
|
||||
Generic single-database configuration.
|
84
api/api/db/alembic/env.py
Normal file
84
api/api/db/alembic/env.py
Normal file
@@ -0,0 +1,84 @@
|
||||
# type: ignore
|
||||
# pylint: disable=all
|
||||
from logging.config import fileConfig
|
||||
|
||||
from sqlalchemy import engine_from_config
|
||||
from sqlalchemy import pool
|
||||
|
||||
from alembic import context
|
||||
|
||||
from api.db import metadata
|
||||
|
||||
# this is the Alembic Config object, which provides
|
||||
# access to the values within the .ini file in use.
|
||||
config = context.config
|
||||
config.set_main_option(
|
||||
"sqlalchemy.url",
|
||||
"mysql+pymysql://root:hackme@localhost:3306/connect_test",
|
||||
)
|
||||
|
||||
# Interpret the config file for Python logging.
|
||||
# This line sets up loggers basically.
|
||||
if config.config_file_name is not None:
|
||||
fileConfig(config.config_file_name)
|
||||
|
||||
# add your model's MetaData object here
|
||||
# for 'autogenerate' support
|
||||
# from myapp import mymodel
|
||||
# target_metadata = mymodel.Base.metadata
|
||||
target_metadata = metadata
|
||||
|
||||
# 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()
|
28
api/api/db/alembic/script.py.mako
Normal file
28
api/api/db/alembic/script.py.mako
Normal file
@@ -0,0 +1,28 @@
|
||||
"""${message}
|
||||
|
||||
Revision ID: ${up_revision}
|
||||
Revises: ${down_revision | comma,n}
|
||||
Create Date: ${create_date}
|
||||
|
||||
"""
|
||||
from typing import Sequence, Union
|
||||
|
||||
from alembic import op
|
||||
import sqlalchemy as sa
|
||||
${imports if imports else ""}
|
||||
|
||||
# revision identifiers, used by Alembic.
|
||||
revision: str = ${repr(up_revision)}
|
||||
down_revision: Union[str, None] = ${repr(down_revision)}
|
||||
branch_labels: Union[str, Sequence[str], None] = ${repr(branch_labels)}
|
||||
depends_on: Union[str, Sequence[str], None] = ${repr(depends_on)}
|
||||
|
||||
|
||||
def upgrade() -> None:
|
||||
"""Upgrade schema."""
|
||||
${upgrades if upgrades else "pass"}
|
||||
|
||||
|
||||
def downgrade() -> None:
|
||||
"""Downgrade schema."""
|
||||
${downgrades if downgrades else "pass"}
|
0
api/api/db/connection/__init__.py
Normal file
0
api/api/db/connection/__init__.py
Normal file
0
api/api/db/logic/__init__.py
Normal file
0
api/api/db/logic/__init__.py
Normal file
0
api/api/db/tables/__init__.py
Normal file
0
api/api/db/tables/__init__.py
Normal file
5
api/api/endpoints/__init__.py
Normal file
5
api/api/endpoints/__init__.py
Normal file
@@ -0,0 +1,5 @@
|
||||
list_of_routes = []
|
||||
|
||||
__all__ = [
|
||||
"list_of_routes",
|
||||
]
|
0
api/api/schemas/__init__.py
Normal file
0
api/api/schemas/__init__.py
Normal file
0
api/api/services/__init__.py
Normal file
0
api/api/services/__init__.py
Normal file
0
api/api/utils/__init__.py
Normal file
0
api/api/utils/__init__.py
Normal file
5
api/api/utils/common.py
Normal file
5
api/api/utils/common.py
Normal file
@@ -0,0 +1,5 @@
|
||||
from urllib.parse import urlparse
|
||||
|
||||
|
||||
def get_hostname(url: str) -> str:
|
||||
return urlparse(url).netloc
|
1237
api/poetry.lock
generated
Normal file
1237
api/poetry.lock
generated
Normal file
File diff suppressed because it is too large
Load Diff
24
api/pyproject.toml
Normal file
24
api/pyproject.toml
Normal file
@@ -0,0 +1,24 @@
|
||||
[project]
|
||||
name = "api"
|
||||
version = "0.1.0"
|
||||
description = ""
|
||||
authors = [
|
||||
{name = "Vladislav",email = "vlad.dev@heado.ru"}
|
||||
]
|
||||
readme = "README.md"
|
||||
requires-python = ">=3.11,<4.0"
|
||||
dependencies = [
|
||||
"sqlalchemy[pymysql,aiomysql] (>=2.0.39,<3.0.0)",
|
||||
"alembic (>=1.15.1,<2.0.0)",
|
||||
"aio-pika (>=9.5.5,<10.0.0)",
|
||||
"fastapi[standart] (>=0.115.11,<0.116.0)",
|
||||
"uvicorn (>=0.34.0,<0.35.0)",
|
||||
"loguru (>=0.7.3,<0.8.0)",
|
||||
"pydantic-settings (>=2.8.1,<3.0.0)",
|
||||
"cryptography (>=44.0.2,<45.0.0)",
|
||||
]
|
||||
|
||||
|
||||
[build-system]
|
||||
requires = ["poetry-core>=2.0.0,<3.0.0"]
|
||||
build-backend = "poetry.core.masonry.api"
|
Reference in New Issue
Block a user