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.
54 lines
1.6 KiB
Python
54 lines
1.6 KiB
Python
from typing import Literal
|
|
|
|
from pydantic import BaseModel, EmailStr, Field
|
|
|
|
|
|
class UploadInitIn(BaseModel):
|
|
filename: str = Field(min_length=1, max_length=255)
|
|
mimeType: str = Field(pattern=r"^(image|video)/[a-z0-9.+-]+$")
|
|
sizeBytes: int = Field(gt=0)
|
|
durationSeconds: int | None = Field(default=None, gt=0)
|
|
authorName: str | None = Field(default=None, max_length=80)
|
|
message: str | None = Field(default=None, max_length=2000)
|
|
|
|
|
|
class MultipartPart(BaseModel):
|
|
partNumber: int = Field(gt=0)
|
|
etag: str = Field(min_length=1)
|
|
|
|
|
|
class MultipartConfirm(BaseModel):
|
|
providerUploadId: str
|
|
parts: list[MultipartPart]
|
|
|
|
|
|
class UploadConfirmIn(BaseModel):
|
|
multipart: MultipartConfirm | None = None
|
|
|
|
|
|
class LoginIn(BaseModel):
|
|
email: EmailStr
|
|
password: str = Field(min_length=1)
|
|
|
|
|
|
class EventConfigUpdate(BaseModel):
|
|
coupleNames: str | None = Field(default=None, min_length=1, max_length=120)
|
|
eventDate: str | None = None
|
|
coverKey: str | None = None
|
|
galleryVisibility: Literal["public", "private"] | None = None
|
|
moderation: Literal["pre", "post"] | None = None
|
|
maxFileMb: int | None = Field(default=None, gt=0)
|
|
allowVideo: bool | None = None
|
|
maxVideoSeconds: int | None = Field(default=None, gt=0)
|
|
welcomeMessage: str | None = Field(default=None, max_length=2000)
|
|
|
|
|
|
class AdminUploadUpdate(BaseModel):
|
|
authorName: str | None = Field(default=None, max_length=80)
|
|
message: str | None = Field(default=None, max_length=2000)
|
|
|
|
|
|
class AdminBulk(BaseModel):
|
|
action: Literal["approve", "reject", "delete"]
|
|
ids: list[str] = Field(min_length=1, max_length=200)
|