Law/alembic/versions/0015_add_clients_table_and_links.py
2026-02-25 18:18:05 +03:00

106 lines
3.7 KiB
Python

"""add clients table and links from requests/invoices
Revision ID: 0015_clients_table_links
Revises: 0014_security_audit_log
Create Date: 2026-02-24
"""
from __future__ import annotations
import uuid
from alembic import op
import sqlalchemy as sa
from sqlalchemy.dialects import postgresql
revision = "0015_clients_table_links"
down_revision = "0014_security_audit_log"
branch_labels = None
depends_on = None
def upgrade():
op.create_table(
"clients",
sa.Column("id", postgresql.UUID(as_uuid=True), primary_key=True),
sa.Column("created_at", sa.DateTime(timezone=True), nullable=True),
sa.Column("updated_at", sa.DateTime(timezone=True), nullable=True),
sa.Column("responsible", sa.String(length=200), nullable=False, server_default="Администратор системы"),
sa.Column("full_name", sa.String(length=200), nullable=False),
sa.Column("phone", sa.String(length=30), nullable=False, unique=True),
)
op.create_index("ix_clients_phone", "clients", ["phone"], unique=True)
op.add_column("requests", sa.Column("client_id", postgresql.UUID(as_uuid=True), nullable=True))
op.add_column("invoices", sa.Column("client_id", postgresql.UUID(as_uuid=True), nullable=True))
op.create_index("ix_requests_client_id", "requests", ["client_id"])
op.create_index("ix_invoices_client_id", "invoices", ["client_id"])
op.create_foreign_key("fk_requests_client_id_clients", "requests", "clients", ["client_id"], ["id"])
op.create_foreign_key("fk_invoices_client_id_clients", "invoices", "clients", ["client_id"], ["id"])
bind = op.get_bind()
rows = bind.execute(
sa.text(
"""
SELECT DISTINCT ON (client_phone)
client_phone,
client_name
FROM requests
WHERE client_phone IS NOT NULL AND client_phone <> ''
ORDER BY client_phone, created_at ASC NULLS FIRST
"""
)
).mappings()
for row in rows:
client_id = uuid.uuid4()
bind.execute(
sa.text(
"""
INSERT INTO clients (id, created_at, updated_at, responsible, full_name, phone)
VALUES (:id, now(), now(), :responsible, :full_name, :phone)
ON CONFLICT (phone) DO NOTHING
"""
),
{
"id": client_id,
"responsible": "Миграция системы",
"full_name": str(row.get("client_name") or "").strip() or "Клиент",
"phone": str(row.get("client_phone") or "").strip(),
},
)
bind.execute(
sa.text(
"""
UPDATE requests AS r
SET client_id = c.id
FROM clients AS c
WHERE r.client_phone = c.phone
AND (r.client_id IS NULL OR r.client_id <> c.id)
"""
)
)
bind.execute(
sa.text(
"""
UPDATE invoices AS i
SET client_id = r.client_id
FROM requests AS r
WHERE i.request_id = r.id
AND (i.client_id IS NULL OR i.client_id <> r.client_id)
"""
)
)
def downgrade():
op.drop_constraint("fk_invoices_client_id_clients", "invoices", type_="foreignkey")
op.drop_constraint("fk_requests_client_id_clients", "requests", type_="foreignkey")
op.drop_index("ix_invoices_client_id", table_name="invoices")
op.drop_index("ix_requests_client_id", table_name="requests")
op.drop_column("invoices", "client_id")
op.drop_column("requests", "client_id")
op.drop_index("ix_clients_phone", table_name="clients")
op.drop_table("clients")