add mock data

This commit is contained in:
TronoSfera 2026-03-02 19:53:42 +03:00
parent 5d63b37928
commit 7785f5c617
2 changed files with 184 additions and 1 deletions

View file

@ -1,7 +1,8 @@
.PHONY: \ .PHONY: \
help \ help \
local-up local-down local-logs local-migrate local-test local-seed \ local-up local-down local-logs local-migrate local-test local-seed local-seed-statuses local-seed-catalog \
prod-up prod-down prod-logs prod-ps prod-migrate \ prod-up prod-down prod-logs prod-ps prod-migrate \
prod-seed-statuses prod-seed-catalog \
prod-secrets-generate prod-secrets-apply prod-secrets-generate-env prod-secrets-apply-env \ prod-secrets-generate prod-secrets-apply prod-secrets-generate-env prod-secrets-apply-env \
prod-minio-tls-init incident-checklist rotate-encryption-kid reencrypt-active-kid \ prod-minio-tls-init incident-checklist rotate-encryption-kid reencrypt-active-kid \
security-smoke prod-security-audit prod-security-scheduler-up prod-security-scheduler-logs \ security-smoke prod-security-audit prod-security-scheduler-up prod-security-scheduler-logs \
@ -34,11 +35,15 @@ help:
@echo " local-migrate - Apply migrations (local)" @echo " local-migrate - Apply migrations (local)"
@echo " local-test - Run backend tests (local)" @echo " local-test - Run backend tests (local)"
@echo " local-seed - Seed quotes (local)" @echo " local-seed - Seed quotes (local)"
@echo " local-seed-statuses - Seed legal flow statuses (local)"
@echo " local-seed-catalog - Seed quotes + legal flow statuses (local)"
@echo " prod-up - Start production stack (nginx 80/443 + TLS certs already issued)" @echo " prod-up - Start production stack (nginx 80/443 + TLS certs already issued)"
@echo " prod-down - Stop production stack" @echo " prod-down - Stop production stack"
@echo " prod-logs - Tail production logs" @echo " prod-logs - Tail production logs"
@echo " prod-ps - Show production services" @echo " prod-ps - Show production services"
@echo " prod-migrate - Apply migrations (prod)" @echo " prod-migrate - Apply migrations (prod)"
@echo " prod-seed-statuses - Seed legal flow statuses (prod)"
@echo " prod-seed-catalog - Seed quotes + legal flow statuses (prod)"
@echo " prod-secrets-generate - Generate rotated internal secrets into .env.prod" @echo " prod-secrets-generate - Generate rotated internal secrets into .env.prod"
@echo " prod-secrets-apply - Generate + apply rotated internal secrets to running prod stack" @echo " prod-secrets-apply - Generate + apply rotated internal secrets to running prod stack"
@echo " prod-secrets-generate-env - Generate rotated secrets from current .env into .env.secure" @echo " prod-secrets-generate-env - Generate rotated secrets from current .env into .env.secure"
@ -83,6 +88,13 @@ local-test:
local-seed: local-seed:
$(LOCAL_COMPOSE) exec -T backend python -m app.scripts.upsert_quotes $(LOCAL_COMPOSE) exec -T backend python -m app.scripts.upsert_quotes
local-seed-statuses:
$(LOCAL_COMPOSE) exec -T backend python -m app.scripts.upsert_statuses_legal_flow
local-seed-catalog:
$(LOCAL_COMPOSE) exec -T backend python -m app.scripts.upsert_quotes
$(LOCAL_COMPOSE) exec -T backend python -m app.scripts.upsert_statuses_legal_flow
check-prod-files: check-prod-files:
@test -f docker-compose.prod.nginx.yml || (echo "[ERROR] Missing docker-compose.prod.nginx.yml. Run: git pull"; exit 1) @test -f docker-compose.prod.nginx.yml || (echo "[ERROR] Missing docker-compose.prod.nginx.yml. Run: git pull"; exit 1)
@test -f frontend/nginx.prod.conf || (echo "[ERROR] Missing frontend/nginx.prod.conf. Run: git pull"; exit 1) @test -f frontend/nginx.prod.conf || (echo "[ERROR] Missing frontend/nginx.prod.conf. Run: git pull"; exit 1)
@ -109,6 +121,13 @@ prod-ps: check-prod-files
prod-migrate: check-prod-files prod-migrate: check-prod-files
$(PROD_COMPOSE) exec -T backend alembic upgrade head $(PROD_COMPOSE) exec -T backend alembic upgrade head
prod-seed-statuses: check-prod-files
$(PROD_COMPOSE) exec -T backend python -m app.scripts.upsert_statuses_legal_flow
prod-seed-catalog: check-prod-files
$(PROD_COMPOSE) exec -T backend python -m app.scripts.upsert_quotes
$(PROD_COMPOSE) exec -T backend python -m app.scripts.upsert_statuses_legal_flow
prod-secrets-generate: prod-secrets-generate:
./scripts/ops/rotate_prod_secrets.sh --env-in .env.production --env-out .env.prod ./scripts/ops/rotate_prod_secrets.sh --env-in .env.production --env-out .env.prod

View file

@ -0,0 +1,164 @@
from __future__ import annotations
from datetime import datetime, timezone
from sqlalchemy.orm import Session
from app.db.session import SessionLocal
from app.models.status import Status
from app.models.status_group import StatusGroup
STATUS_GROUP_NAME = "Юридический процесс"
STATUS_GROUP_SORT_ORDER = 10
RESPONSIBLE = "Импорт статусов (prod)"
LEGAL_FLOW_STATUSES = [
{"code": "PRELIMINARY_CONSULT", "name": "Предварительная консультация", "sort_order": 10, "is_terminal": False},
{"code": "INVOICE_ISSUANCE", "name": "Выставление счета", "sort_order": 20, "is_terminal": False},
{"code": "CONTRACT_DISPATCH", "name": "Направление договора", "sort_order": 30, "is_terminal": False},
{"code": "ADDENDUM_DISPATCH", "name": "Направление допсоглашения", "sort_order": 40, "is_terminal": False},
{"code": "INVOICE_PAYMENT", "name": "Оплата счета", "sort_order": 50, "is_terminal": False},
{"code": "LEGAL_STRATEGY", "name": "Разработка юридической стратегии", "sort_order": 60, "is_terminal": False},
{"code": "NEGOTIATION", "name": "Ведение переговоров", "sort_order": 70, "is_terminal": False},
{"code": "CLAIM_PREPARATION", "name": "Подготовка претензии", "sort_order": 80, "is_terminal": False},
{"code": "CLAIM_DISPATCH", "name": "Претензия направлена", "sort_order": 90, "is_terminal": False},
{"code": "LAWSUIT_PREPARATION", "name": "Подготовка иска", "sort_order": 100, "is_terminal": False},
{"code": "CONTRACT_PREPARATION", "name": "Разработка договора", "sort_order": 110, "is_terminal": False},
{"code": "LEGAL_POSITION_PREPARATION", "name": "Разработка правовой позиции", "sort_order": 120, "is_terminal": False},
{"code": "LAWSUIT_PREPARATION_REPEAT", "name": "Подготовка иска", "sort_order": 130, "is_terminal": False},
{"code": "STATE_FEE_PAYMENT", "name": "Оплата госпошлины", "sort_order": 140, "is_terminal": False},
{"code": "LAWSUIT_FILING", "name": "Подача иска", "sort_order": 150, "is_terminal": False},
{"code": "COURT_HEARING", "name": "Судебное заседание", "sort_order": 160, "is_terminal": False},
{"code": "EXPERT_EXAM_APPOINTMENT", "name": "Назначение экспертизы", "sort_order": 170, "is_terminal": False},
{"code": "FIRST_INSTANCE_DECISION", "name": "Вынесение решения суда первая инстанция", "sort_order": 180, "is_terminal": False},
{
"code": "APPEAL_BRIEF_PREPARATION",
"name": "Подготовка апелляционной жалобы/отзыва",
"sort_order": 190,
"is_terminal": False,
},
{"code": "APPEAL_BRIEF_FILING", "name": "Подача апелляционной жалобы/отзыва", "sort_order": 200, "is_terminal": False},
{"code": "APPEAL_ACT_DECISION", "name": "Вынесение судебного акта апелляция", "sort_order": 210, "is_terminal": False},
{
"code": "CASSATION_BRIEF_PREPARATION",
"name": "Подготовка кассационной жалобы/отзыва",
"sort_order": 220,
"is_terminal": False,
},
{"code": "CASSATION_BRIEF_FILING", "name": "Подача кассационной жалобы/отзыва", "sort_order": 230, "is_terminal": False},
{"code": "CASSATION_ACT_DECISION", "name": "Вынесение судебного акта кассация", "sort_order": 240, "is_terminal": False},
{"code": "SUPREME_COURT_COMPLAINT_FILING", "name": "Подача жалобы в ВС РФ", "sort_order": 250, "is_terminal": False},
{"code": "SUPREME_COURT_REVIEW", "name": "Рассмотрение жалобы в ВС РФ", "sort_order": 260, "is_terminal": False},
{"code": "ENFORCEMENT_PROCEEDINGS", "name": "Исполнительное производство", "sort_order": 270, "is_terminal": False},
{"code": "FINAL_SETTLEMENT", "name": "Окончательный расчет", "sort_order": 280, "is_terminal": False},
{"code": "BONUS_PAYMENT", "name": "Премирование", "sort_order": 290, "is_terminal": True},
]
def ensure_status_group(db: Session) -> StatusGroup:
row = db.query(StatusGroup).filter(StatusGroup.name == STATUS_GROUP_NAME).first()
if row is None:
row = StatusGroup(
name=STATUS_GROUP_NAME,
sort_order=STATUS_GROUP_SORT_ORDER,
responsible=RESPONSIBLE,
)
db.add(row)
db.flush()
return row
changed = False
if int(row.sort_order or 0) != STATUS_GROUP_SORT_ORDER:
row.sort_order = STATUS_GROUP_SORT_ORDER
changed = True
if str(row.responsible or "") != RESPONSIBLE:
row.responsible = RESPONSIBLE
changed = True
if changed:
row.updated_at = datetime.now(timezone.utc)
db.add(row)
return row
def upsert_statuses(db: Session) -> tuple[int, int, int]:
group = ensure_status_group(db)
created = 0
updated = 0
unchanged = 0
for item in LEGAL_FLOW_STATUSES:
code = str(item["code"]).strip()
name = str(item["name"]).strip()
sort_order = int(item["sort_order"])
is_terminal = bool(item["is_terminal"])
row = db.query(Status).filter(Status.code == code).first()
if row is None:
db.add(
Status(
code=code,
name=name,
status_group_id=group.id,
enabled=True,
sort_order=sort_order,
is_terminal=is_terminal,
kind="DEFAULT",
responsible=RESPONSIBLE,
)
)
created += 1
continue
changed = False
if str(row.name or "") != name:
row.name = name
changed = True
if row.status_group_id != group.id:
row.status_group_id = group.id
changed = True
if not bool(row.enabled):
row.enabled = True
changed = True
if int(row.sort_order or 0) != sort_order:
row.sort_order = sort_order
changed = True
if bool(row.is_terminal) != is_terminal:
row.is_terminal = is_terminal
changed = True
if str(row.kind or "").upper() != "DEFAULT":
row.kind = "DEFAULT"
changed = True
if str(row.responsible or "") != RESPONSIBLE:
row.responsible = RESPONSIBLE
changed = True
if changed:
row.updated_at = datetime.now(timezone.utc)
db.add(row)
updated += 1
else:
unchanged += 1
db.commit()
return created, updated, unchanged
def main() -> None:
db = SessionLocal()
try:
created, updated, unchanged = upsert_statuses(db)
total = db.query(Status).count()
finally:
db.close()
print(
"statuses upsert done: "
f"created={created}, updated={updated}, unchanged={unchanged}, "
f"flow_total={len(LEGAL_FLOW_STATUSES)}, total_in_db={total}"
)
if __name__ == "__main__":
main()