awesome-fastapi-projects/app/conftest.py
2023-11-19 23:17:57 +01:00

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