Repo passa a viver dentro de infra/, com três stacks isoladas que
compartilham uma network Docker externa ('infra-net'):
main/ infra base: postgres + redis + minio + minio-init + pgadmin
+ caddy. Postgres roda em modo multi-banco; o init script
cria os DBs 'wedding' e 'gitea' com roles dedicadas. MinIO
tem um bucket inicial criado pelo minio-init com anonymous
download + CORS. Caddy é o único container expondo 80/443
e roteia por hostname: gitea.{DOMAIN_BASE} / wedding.* /
pgadmin.* / minio.* / media.* (rewrite de bucket).
gitea/ gitea + act_runner. Gitea liga no postgres compartilhado e
usa redis pra cache+sessões. O runner ganha um Dockerfile
pequeno que adiciona docker CLI por cima do
gitea/act_runner pra workflows poderem chamar 'docker
build'. Bootstrap do token de runner documentado no
.env.example.
wedding_photo/ Só a aplicação: 'wedding_app' (FastAPI + SPA) +
postgres-backup + media-backup. Os bancos e o MinIO vêm
da stack main/. A app usa extra_hosts: host-gateway pra
alcançar media.{DOMAIN_BASE} via Caddy mesmo em dev local
— assim a assinatura S3 fecha com o host que o browser
usa pra fazer PUT.
Orquestração:
- Makefile no root: 'make up' sobe tudo na ordem (main -> gitea ->
wedding_photo). 'make up-{main,gitea,wedding}' pra controle
granular. 'make logs-*', 'make down', 'make status', 'make pull-*'.
- network.sh cria a 'infra-net' antes de qualquer up; idempotente.
- Cada stack tem seu próprio .env.example. As creds compartilhadas
(DOMAIN_BASE, MINIO_ROOT_*, WEDDING_DB_*) precisam casar entre
main/.env e o consumidor (gitea/.env ou wedding_photo/.env).
- .gitignore ignora todas as pastas data/ dos volumes.
47 lines
1.4 KiB
SQL
47 lines
1.4 KiB
SQL
CREATE TABLE event_config (
|
|
id INTEGER PRIMARY KEY DEFAULT 1,
|
|
couple_names TEXT NOT NULL,
|
|
event_date TEXT,
|
|
cover_key TEXT,
|
|
gallery_visibility TEXT NOT NULL DEFAULT 'public',
|
|
moderation TEXT NOT NULL DEFAULT 'post',
|
|
max_file_mb INTEGER NOT NULL DEFAULT 500,
|
|
allow_video BOOLEAN NOT NULL DEFAULT TRUE,
|
|
max_video_seconds INTEGER NOT NULL DEFAULT 300,
|
|
welcome_message TEXT,
|
|
updated_at BIGINT NOT NULL DEFAULT (EXTRACT(EPOCH FROM NOW()) * 1000)::BIGINT
|
|
);
|
|
|
|
INSERT INTO event_config (id, couple_names)
|
|
VALUES (1, 'Stefanie & Leandro')
|
|
ON CONFLICT (id) DO NOTHING;
|
|
|
|
CREATE TABLE uploads (
|
|
id TEXT PRIMARY KEY,
|
|
storage_key TEXT NOT NULL,
|
|
thumbnail_key TEXT,
|
|
mime_type TEXT NOT NULL,
|
|
size_bytes BIGINT NOT NULL,
|
|
duration_seconds INTEGER,
|
|
author_name TEXT,
|
|
message TEXT,
|
|
status TEXT NOT NULL DEFAULT 'approved',
|
|
source TEXT NOT NULL DEFAULT 'guest',
|
|
ip_hash TEXT,
|
|
provider_upload_id TEXT,
|
|
created_at BIGINT NOT NULL DEFAULT (EXTRACT(EPOCH FROM NOW()) * 1000)::BIGINT,
|
|
approved_at BIGINT
|
|
);
|
|
|
|
CREATE INDEX idx_uploads_status_created ON uploads (status, created_at DESC);
|
|
CREATE INDEX idx_uploads_source ON uploads (source);
|
|
CREATE INDEX idx_uploads_author_lower ON uploads (LOWER(author_name));
|
|
|
|
CREATE TABLE audit_log (
|
|
id TEXT PRIMARY KEY,
|
|
action TEXT NOT NULL,
|
|
actor TEXT,
|
|
payload TEXT,
|
|
created_at BIGINT NOT NULL DEFAULT (EXTRACT(EPOCH FROM NOW()) * 1000)::BIGINT
|
|
);
|