mirror of
https://github.com/Kludex/awesome-fastapi-projects.git
synced 2024-11-27 14:01:09 +00:00
170 lines
5.4 KiB
Python
170 lines
5.4 KiB
Python
"""The application-level conftest."""
|
|
import asyncio
|
|
import contextlib
|
|
from collections.abc import AsyncGenerator, Generator
|
|
from typing import Literal
|
|
|
|
import pytest
|
|
import stamina
|
|
from dirty_equals import IsList
|
|
from sqlalchemy.ext.asyncio import (
|
|
AsyncConnection,
|
|
AsyncEngine,
|
|
AsyncSession,
|
|
async_sessionmaker,
|
|
create_async_engine,
|
|
)
|
|
|
|
from app.database import Dependency, Repo
|
|
from app.factories import DependencyCreateDataFactory
|
|
from app.source_graph.factories import SourceGraphRepoDataFactory
|
|
from app.source_graph.models import SourceGraphRepoData
|
|
|
|
|
|
@pytest.fixture(autouse=True, scope="session")
|
|
def anyio_backend() -> Literal["asyncio"]:
|
|
"""Use asyncio as the async backend."""
|
|
return "asyncio"
|
|
|
|
|
|
@pytest.fixture(autouse=True, scope="session")
|
|
def _deactivate_retries() -> None:
|
|
"""Deactivate stamina retries."""
|
|
stamina.set_active(False)
|
|
|
|
|
|
@pytest.fixture(scope="session")
|
|
def db_path() -> str:
|
|
"""Use the in-memory database for tests."""
|
|
return "" # ":memory:"
|
|
|
|
|
|
@pytest.fixture(scope="session")
|
|
def db_connection_string(
|
|
db_path: str,
|
|
) -> str:
|
|
"""Provide the connection string for the in-memory database."""
|
|
return f"sqlite+aiosqlite:///{db_path}"
|
|
|
|
|
|
@pytest.fixture(scope="session", params=[{"echo": False}], ids=["echo=False"])
|
|
async def db_engine(
|
|
db_connection_string: str,
|
|
request: pytest.FixtureRequest,
|
|
) -> AsyncGenerator[AsyncEngine, None, None]:
|
|
"""Create the database engine."""
|
|
# echo=True enables logging of all SQL statements
|
|
# https://docs.sqlalchemy.org/en/20/core/engines.html#sqlalchemy.create_engine.params.echo
|
|
engine = create_async_engine(
|
|
db_connection_string,
|
|
**request.param, # type: ignore
|
|
)
|
|
try:
|
|
yield engine
|
|
finally:
|
|
# for AsyncEngine created in function scope, close and
|
|
# clean-up pooled connections
|
|
await engine.dispose()
|
|
|
|
|
|
@pytest.fixture(scope="session")
|
|
def event_loop(
|
|
request: pytest.FixtureRequest,
|
|
) -> Generator[asyncio.AbstractEventLoop, None, None]:
|
|
"""
|
|
Create an instance of the default event loop for a session.
|
|
|
|
An event loop is destroyed at the end of the test session.
|
|
https://docs.pytest.org/en/6.2.x/fixture.html#fixture-scopes
|
|
"""
|
|
with contextlib.closing(loop := asyncio.get_event_loop_policy().get_event_loop()):
|
|
yield loop
|
|
|
|
|
|
@pytest.fixture(scope="session")
|
|
async def _database_objects(
|
|
db_engine: AsyncEngine,
|
|
) -> AsyncGenerator[None, None]:
|
|
"""Create the database objects (tables, etc.)."""
|
|
from app.database import Base
|
|
|
|
# Enters a transaction
|
|
# https://docs.sqlalchemy.org/en/20/orm/extensions/asyncio.html#sqlalchemy.ext.asyncio.AsyncConnection.begin
|
|
try:
|
|
async with db_engine.begin() as conn:
|
|
await conn.run_sync(Base.metadata.drop_all)
|
|
await conn.run_sync(Base.metadata.create_all)
|
|
yield
|
|
finally:
|
|
# Clean up after the testing session is over
|
|
async with db_engine.begin() as conn:
|
|
await conn.run_sync(Base.metadata.drop_all)
|
|
|
|
|
|
@pytest.fixture(scope="session")
|
|
async def db_connection(
|
|
db_engine: AsyncEngine,
|
|
) -> AsyncGenerator[AsyncConnection, None]:
|
|
"""Create a database connection."""
|
|
# Return connection with no transaction
|
|
# https://docs.sqlalchemy.org/en/20/orm/extensions/asyncio.html#sqlalchemy.ext.asyncio.AsyncEngine.connect
|
|
async with db_engine.connect() as conn:
|
|
yield conn
|
|
|
|
|
|
@pytest.fixture()
|
|
async def db_session(
|
|
db_engine: AsyncEngine,
|
|
_database_objects: None,
|
|
) -> AsyncGenerator[AsyncSession, None]:
|
|
"""Create a database session."""
|
|
# The `async_sessionmaker` function is used to create a Session factory
|
|
# https://docs.sqlalchemy.org/en/20/orm/extensions/asyncio.html#sqlalchemy.ext.asyncio.async_sessionmaker
|
|
async_session_factory = async_sessionmaker(
|
|
db_engine, expire_on_commit=False, autoflush=False, autocommit=False
|
|
)
|
|
async with async_session_factory() as session:
|
|
yield session
|
|
|
|
|
|
@pytest.fixture()
|
|
async def db_uow(
|
|
db_session: AsyncSession,
|
|
) -> AsyncGenerator[AsyncSession, None]:
|
|
"""Provide a transactional scope around a series of operations."""
|
|
from app.uow import async_session_uow
|
|
|
|
# This context manager will start a transaction, and roll it back at the end
|
|
# https://docs.sqlalchemy.org/en/20/orm/extensions/asyncio.html#sqlalchemy.ext.asyncio.AsyncSessionTransaction
|
|
async with async_session_uow(db_session) as session:
|
|
yield session
|
|
|
|
|
|
@pytest.fixture()
|
|
async def some_repos(
|
|
db_session: AsyncSession,
|
|
source_graph_repo_data_factory: SourceGraphRepoDataFactory,
|
|
dependency_create_data_factory: DependencyCreateDataFactory,
|
|
) -> list[Repo]:
|
|
"""Create some repos."""
|
|
source_graph_repos_data: list[
|
|
SourceGraphRepoData
|
|
] = source_graph_repo_data_factory.batch(10)
|
|
assert source_graph_repos_data == IsList(length=10)
|
|
repos = [
|
|
Repo(
|
|
url=str(source_graph_repo_data.repo_url),
|
|
description=source_graph_repo_data.description,
|
|
stars=source_graph_repo_data.stars,
|
|
source_graph_repo_id=source_graph_repo_data.repo_id,
|
|
dependencies=[
|
|
Dependency(**dependency_create_data.model_dump())
|
|
for dependency_create_data in dependency_create_data_factory.batch(5)
|
|
],
|
|
)
|
|
for source_graph_repo_data in source_graph_repos_data
|
|
]
|
|
db_session.add_all(repos)
|
|
await db_session.flush()
|
|
return repos
|