feat: add approval-task links model and related functionality for task associations

This commit is contained in:
Abhimanyu Saharan
2026-02-11 20:27:04 +05:30
parent 3dfdfa3c3e
commit af8a263c27
19 changed files with 870 additions and 129 deletions

View File

@@ -0,0 +1,137 @@
"""add approval task links
Revision ID: f4d2b649e93a
Revises: c3b58a391f2e
Create Date: 2026-02-11 20:05:00.000000
"""
from __future__ import annotations
from uuid import uuid4
import sqlalchemy as sa
from alembic import op
# revision identifiers, used by Alembic.
revision = "f4d2b649e93a"
down_revision = "c3b58a391f2e"
branch_labels = None
depends_on = None
def upgrade() -> None:
bind = op.get_bind()
inspector = sa.inspect(bind)
if not inspector.has_table("approval_task_links"):
op.create_table(
"approval_task_links",
sa.Column("id", sa.Uuid(), nullable=False),
sa.Column("approval_id", sa.Uuid(), nullable=False),
sa.Column("task_id", sa.Uuid(), nullable=False),
sa.Column("created_at", sa.DateTime(), nullable=False),
sa.ForeignKeyConstraint(["approval_id"], ["approvals.id"]),
sa.ForeignKeyConstraint(["task_id"], ["tasks.id"]),
sa.PrimaryKeyConstraint("id"),
sa.UniqueConstraint(
"approval_id",
"task_id",
name="uq_approval_task_links_approval_id_task_id",
),
)
else:
target_unique_columns = ("approval_id", "task_id")
unique_constraints = inspector.get_unique_constraints("approval_task_links")
has_target_unique = False
for item in unique_constraints:
columns = tuple(item.get("column_names") or ())
if columns == target_unique_columns:
has_target_unique = True
break
if not has_target_unique:
op.create_unique_constraint(
"uq_approval_task_links_approval_id_task_id",
"approval_task_links",
["approval_id", "task_id"],
)
indexes = inspector.get_indexes("approval_task_links")
has_approval_id_index = any(
tuple(item.get("column_names") or ()) == ("approval_id",) for item in indexes
)
has_task_id_index = any(tuple(item.get("column_names") or ()) == ("task_id",) for item in indexes)
if not has_approval_id_index:
op.create_index(
op.f("ix_approval_task_links_approval_id"),
"approval_task_links",
["approval_id"],
unique=False,
)
if not has_task_id_index:
op.create_index(
op.f("ix_approval_task_links_task_id"),
"approval_task_links",
["task_id"],
unique=False,
)
link_table = sa.table(
"approval_task_links",
sa.column("id", sa.Uuid()),
sa.column("approval_id", sa.Uuid()),
sa.column("task_id", sa.Uuid()),
sa.column("created_at", sa.DateTime()),
)
approvals_table = sa.table(
"approvals",
sa.column("id", sa.Uuid()),
sa.column("task_id", sa.Uuid()),
sa.column("created_at", sa.DateTime()),
)
rows = list(
bind.execute(
sa.select(
approvals_table.c.id,
approvals_table.c.task_id,
approvals_table.c.created_at,
)
.select_from(approvals_table)
.where(approvals_table.c.task_id.is_not(None)),
),
)
existing_links = {
(approval_id, task_id)
for approval_id, task_id in list(
bind.execute(
sa.select(
sa.column("approval_id"),
sa.column("task_id"),
).select_from(sa.table("approval_task_links")),
),
)
}
missing_rows = [
(approval_id, task_id, created_at)
for approval_id, task_id, created_at in rows
if (approval_id, task_id) not in existing_links
]
if missing_rows:
op.bulk_insert(
link_table,
[
{
"id": uuid4(),
"approval_id": approval_id,
"task_id": task_id,
"created_at": created_at,
}
for approval_id, task_id, created_at in missing_rows
],
)
def downgrade() -> None:
op.drop_index(op.f("ix_approval_task_links_task_id"), table_name="approval_task_links")
op.drop_index(op.f("ix_approval_task_links_approval_id"), table_name="approval_task_links")
op.drop_table("approval_task_links")