generated from coulomb/repo-seed
Schema (src/artifactstore/db/schema.py): - events table (ADR-0002 source of truth): sequence BIGSERIAL PK, created_at, event_type, subject_kind, subject_id, actor, payload (CBOR bytes), payload_digest. Indexes on (subject_kind, subject_id) and (event_type, sequence). - artifact_packages, artifact_files, storage_locations, retention_state (materialised views over events). - retention_classes (seed table) and metadata_schemas (config table). - ADR-0001 columns present: digest_algorithm, digest_primary, digest_sha256, content_address. Blueprint tiering columns present: retrieval_tier (default 'hot'), restore_status. - Types portable: SQLAlchemy 2.0 Core with JSON().with_variant(JSONB, 'postgresql'), Uuid, LargeBinary, DateTime(timezone=True), Boolean false() default. Seed (src/artifactstore/db/seed.py): five v1 retention classes (transient, raw-evidence, summary-evidence, release-evidence, permanent-record) with default durations in seconds; permanent-record has no expiry. Alembic: - alembic.ini with sync sqlite URL default; path_separator=os to silence the 1.13 deprecation warning. - migrations/env.py: translates async URLs (+aiosqlite, +asyncpg) to sync counterparts at migrate-time so a single ARTIFACTSTORE_DATABASE_URL works for both runtime (async) and Alembic (sync). - migrations/script.py.mako template. - migrations/versions/20260516_0001_initial.py: metadata.create_all + bulk insert of retention class seeds. Make: - make migrate: alembic upgrade head (ensures var/ exists). - make migrate-fresh: drop local SQLite + re-run. Deps: psycopg[binary] added as optional `postgres` extra (PostgreSQL prod path; SQLite default for dev needs no extra). Tests: - tests/unit/test_db_schema.py: every expected table present; ADR-0001 and tiering columns present; seed has the five v1 classes; permanent-record has no default_duration; create_all + FK insert + Boolean default round-trip on in-memory SQLite. - tests/integration/test_migrations.py: alembic upgrade head against a tempfile SQLite produces all tables (+ alembic_version) and the seed rows. Gates: ruff clean, mypy --strict clean on 32 files, 38 tests pass. Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
108 lines
2.3 KiB
TOML
108 lines
2.3 KiB
TOML
[build-system]
|
|
requires = ["hatchling>=1.25"]
|
|
build-backend = "hatchling.build"
|
|
|
|
[project]
|
|
name = "artifactstore"
|
|
version = "0.1.0"
|
|
description = "Generic artifact registry and storage gateway"
|
|
readme = "README.md"
|
|
requires-python = ">=3.12"
|
|
license = { file = "LICENSE" }
|
|
authors = [{ name = "artifact-store contributors" }]
|
|
keywords = ["artifact", "storage", "registry", "evidence", "retention"]
|
|
classifiers = [
|
|
"Development Status :: 2 - Pre-Alpha",
|
|
"Intended Audience :: Developers",
|
|
"Operating System :: POSIX :: Linux",
|
|
"Programming Language :: Python :: 3.12",
|
|
"Programming Language :: Python :: 3.13",
|
|
"Topic :: Software Development :: Libraries",
|
|
"Topic :: System :: Archiving",
|
|
]
|
|
dependencies = [
|
|
"fastapi >= 0.115",
|
|
"uvicorn[standard] >= 0.30",
|
|
"sqlalchemy >= 2.0",
|
|
"asyncpg >= 0.29",
|
|
"aiosqlite >= 0.20",
|
|
"alembic >= 1.13",
|
|
"blake3 >= 0.4",
|
|
"cbor2 >= 5.6",
|
|
"jcs >= 0.2",
|
|
"typer >= 0.12",
|
|
"structlog >= 24.1",
|
|
"pydantic >= 2.7",
|
|
"pydantic-settings >= 2.4",
|
|
]
|
|
|
|
[project.optional-dependencies]
|
|
dev = [
|
|
"pytest >= 8.0",
|
|
"pytest-asyncio >= 0.23",
|
|
"httpx >= 0.27",
|
|
"hypothesis >= 6.100",
|
|
"ruff >= 0.6",
|
|
"mypy >= 1.10",
|
|
]
|
|
postgres = [
|
|
"psycopg[binary] >= 3.2",
|
|
]
|
|
|
|
[project.scripts]
|
|
artifactstore = "artifactstore.cli:app"
|
|
|
|
[tool.hatch.build.targets.wheel]
|
|
packages = ["src/artifactstore"]
|
|
|
|
[tool.uv]
|
|
dev-dependencies = [
|
|
"pytest >= 8.0",
|
|
"pytest-asyncio >= 0.23",
|
|
"httpx >= 0.27",
|
|
"hypothesis >= 6.100",
|
|
"ruff >= 0.6",
|
|
"mypy >= 1.10",
|
|
]
|
|
|
|
[tool.ruff]
|
|
target-version = "py312"
|
|
line-length = 100
|
|
src = ["src", "tests"]
|
|
extend-exclude = [".venv", "var", "migrations"]
|
|
|
|
[tool.ruff.lint]
|
|
select = ["E", "F", "W", "I", "N", "UP", "B", "C4", "RET", "SIM", "RUF"]
|
|
ignore = [
|
|
"B008", # FastAPI dependency injection idiom
|
|
]
|
|
|
|
[tool.ruff.lint.per-file-ignores]
|
|
"tests/**" = ["S101"]
|
|
"src/artifactstore/cli/__init__.py" = ["UP007"]
|
|
|
|
[tool.ruff.format]
|
|
quote-style = "double"
|
|
indent-style = "space"
|
|
|
|
[tool.mypy]
|
|
python_version = "3.12"
|
|
strict = true
|
|
files = ["src", "tests"]
|
|
mypy_path = "src"
|
|
explicit_package_bases = true
|
|
namespace_packages = true
|
|
|
|
[[tool.mypy.overrides]]
|
|
module = ["jcs"]
|
|
ignore_missing_imports = true
|
|
|
|
[tool.pytest.ini_options]
|
|
asyncio_mode = "auto"
|
|
testpaths = ["tests"]
|
|
pythonpath = ["src"]
|
|
addopts = "-q --strict-markers"
|
|
markers = [
|
|
"integration: marks tests as integration (requires DB / backend)",
|
|
]
|