Database & Migrations: - Update all Alembic migrations for PostgreSQL compatibility - Remove SQLite-specific syntax (AUTOINCREMENT, etc.) - Add database utility helpers for PostgreSQL operations - Fix services to use PostgreSQL-compatible queries Documentation: - Add comprehensive Docker deployment guide - Add production deployment documentation - Add infrastructure architecture documentation - Update database setup guide for PostgreSQL-only - Expand troubleshooting guide Architecture & Validation: - Add migration.yaml rules for SQL compatibility checking - Enhance validate_architecture.py with migration validation - Update architecture rules to validate Alembic migrations Development: - Fix duplicate install-all target in Makefile - Add Celery/Redis validation to install.py script - Add docker-compose.test.yml for CI testing - Add squash_migrations.py utility script - Update tests for PostgreSQL compatibility - Improve test fixtures in conftest.py Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
292 lines
9.5 KiB
Python
292 lines
9.5 KiB
Python
"""add_architecture_quality_tracking_tables
|
|
|
|
Revision ID: 7a7ce92593d5
|
|
Revises: a2064e1dfcd4
|
|
Create Date: 2025-11-28 09:21:16.545203
|
|
|
|
"""
|
|
|
|
from typing import Sequence, Union
|
|
|
|
import sqlalchemy as sa
|
|
from sqlalchemy.dialects import postgresql
|
|
|
|
from alembic import op
|
|
|
|
# revision identifiers, used by Alembic.
|
|
revision: str = "7a7ce92593d5"
|
|
down_revision: Union[str, None] = "a2064e1dfcd4"
|
|
branch_labels: Union[str, Sequence[str], None] = None
|
|
depends_on: Union[str, Sequence[str], None] = None
|
|
|
|
|
|
def upgrade() -> None:
|
|
# Create architecture_scans table
|
|
op.create_table(
|
|
"architecture_scans",
|
|
sa.Column("id", sa.Integer(), nullable=False),
|
|
sa.Column(
|
|
"timestamp",
|
|
sa.DateTime(timezone=True),
|
|
server_default=sa.text("CURRENT_TIMESTAMP"),
|
|
nullable=False,
|
|
),
|
|
sa.Column("total_files", sa.Integer(), nullable=True),
|
|
sa.Column("total_violations", sa.Integer(), nullable=True),
|
|
sa.Column("errors", sa.Integer(), nullable=True),
|
|
sa.Column("warnings", sa.Integer(), nullable=True),
|
|
sa.Column("duration_seconds", sa.Float(), nullable=True),
|
|
sa.Column("triggered_by", sa.String(length=100), nullable=True),
|
|
sa.Column("git_commit_hash", sa.String(length=40), nullable=True),
|
|
sa.PrimaryKeyConstraint("id"),
|
|
)
|
|
op.create_index(
|
|
op.f("ix_architecture_scans_id"), "architecture_scans", ["id"], unique=False
|
|
)
|
|
op.create_index(
|
|
op.f("ix_architecture_scans_timestamp"),
|
|
"architecture_scans",
|
|
["timestamp"],
|
|
unique=False,
|
|
)
|
|
|
|
# Create architecture_rules table
|
|
op.create_table(
|
|
"architecture_rules",
|
|
sa.Column("id", sa.Integer(), nullable=False),
|
|
sa.Column("rule_id", sa.String(length=20), nullable=False),
|
|
sa.Column("category", sa.String(length=50), nullable=False),
|
|
sa.Column("name", sa.String(length=200), nullable=False),
|
|
sa.Column("description", sa.Text(), nullable=True),
|
|
sa.Column("severity", sa.String(length=10), nullable=False),
|
|
sa.Column("enabled", sa.Boolean(), nullable=False, server_default="1"),
|
|
sa.Column("custom_config", sa.JSON(), nullable=True),
|
|
sa.Column(
|
|
"created_at",
|
|
sa.DateTime(timezone=True),
|
|
server_default=sa.text("CURRENT_TIMESTAMP"),
|
|
nullable=False,
|
|
),
|
|
sa.Column(
|
|
"updated_at",
|
|
sa.DateTime(timezone=True),
|
|
server_default=sa.text("CURRENT_TIMESTAMP"),
|
|
nullable=False,
|
|
),
|
|
sa.PrimaryKeyConstraint("id"),
|
|
sa.UniqueConstraint("rule_id"),
|
|
)
|
|
op.create_index(
|
|
op.f("ix_architecture_rules_id"), "architecture_rules", ["id"], unique=False
|
|
)
|
|
op.create_index(
|
|
op.f("ix_architecture_rules_rule_id"),
|
|
"architecture_rules",
|
|
["rule_id"],
|
|
unique=True,
|
|
)
|
|
|
|
# Create architecture_violations table
|
|
op.create_table(
|
|
"architecture_violations",
|
|
sa.Column("id", sa.Integer(), nullable=False),
|
|
sa.Column("scan_id", sa.Integer(), nullable=False),
|
|
sa.Column("rule_id", sa.String(length=20), nullable=False),
|
|
sa.Column("rule_name", sa.String(length=200), nullable=False),
|
|
sa.Column("severity", sa.String(length=10), nullable=False),
|
|
sa.Column("file_path", sa.String(length=500), nullable=False),
|
|
sa.Column("line_number", sa.Integer(), nullable=False),
|
|
sa.Column("message", sa.Text(), nullable=False),
|
|
sa.Column("context", sa.Text(), nullable=True),
|
|
sa.Column("suggestion", sa.Text(), nullable=True),
|
|
sa.Column("status", sa.String(length=20), server_default="open", nullable=True),
|
|
sa.Column("assigned_to", sa.Integer(), nullable=True),
|
|
sa.Column("resolved_at", sa.DateTime(timezone=True), nullable=True),
|
|
sa.Column("resolved_by", sa.Integer(), nullable=True),
|
|
sa.Column("resolution_note", sa.Text(), nullable=True),
|
|
sa.Column(
|
|
"created_at",
|
|
sa.DateTime(timezone=True),
|
|
server_default=sa.text("CURRENT_TIMESTAMP"),
|
|
nullable=False,
|
|
),
|
|
sa.ForeignKeyConstraint(
|
|
["assigned_to"],
|
|
["users.id"],
|
|
),
|
|
sa.ForeignKeyConstraint(
|
|
["resolved_by"],
|
|
["users.id"],
|
|
),
|
|
sa.ForeignKeyConstraint(
|
|
["scan_id"],
|
|
["architecture_scans.id"],
|
|
),
|
|
sa.PrimaryKeyConstraint("id"),
|
|
)
|
|
op.create_index(
|
|
op.f("ix_architecture_violations_file_path"),
|
|
"architecture_violations",
|
|
["file_path"],
|
|
unique=False,
|
|
)
|
|
op.create_index(
|
|
op.f("ix_architecture_violations_id"),
|
|
"architecture_violations",
|
|
["id"],
|
|
unique=False,
|
|
)
|
|
op.create_index(
|
|
op.f("ix_architecture_violations_rule_id"),
|
|
"architecture_violations",
|
|
["rule_id"],
|
|
unique=False,
|
|
)
|
|
op.create_index(
|
|
op.f("ix_architecture_violations_scan_id"),
|
|
"architecture_violations",
|
|
["scan_id"],
|
|
unique=False,
|
|
)
|
|
op.create_index(
|
|
op.f("ix_architecture_violations_severity"),
|
|
"architecture_violations",
|
|
["severity"],
|
|
unique=False,
|
|
)
|
|
op.create_index(
|
|
op.f("ix_architecture_violations_status"),
|
|
"architecture_violations",
|
|
["status"],
|
|
unique=False,
|
|
)
|
|
|
|
# Create violation_assignments table
|
|
op.create_table(
|
|
"violation_assignments",
|
|
sa.Column("id", sa.Integer(), nullable=False),
|
|
sa.Column("violation_id", sa.Integer(), nullable=False),
|
|
sa.Column("user_id", sa.Integer(), nullable=False),
|
|
sa.Column(
|
|
"assigned_at",
|
|
sa.DateTime(timezone=True),
|
|
server_default=sa.text("CURRENT_TIMESTAMP"),
|
|
nullable=False,
|
|
),
|
|
sa.Column("assigned_by", sa.Integer(), nullable=True),
|
|
sa.Column("due_date", sa.DateTime(timezone=True), nullable=True),
|
|
sa.Column(
|
|
"priority", sa.String(length=10), server_default="medium", nullable=True
|
|
),
|
|
sa.ForeignKeyConstraint(
|
|
["assigned_by"],
|
|
["users.id"],
|
|
),
|
|
sa.ForeignKeyConstraint(
|
|
["user_id"],
|
|
["users.id"],
|
|
),
|
|
sa.ForeignKeyConstraint(
|
|
["violation_id"],
|
|
["architecture_violations.id"],
|
|
),
|
|
sa.PrimaryKeyConstraint("id"),
|
|
)
|
|
op.create_index(
|
|
op.f("ix_violation_assignments_id"),
|
|
"violation_assignments",
|
|
["id"],
|
|
unique=False,
|
|
)
|
|
op.create_index(
|
|
op.f("ix_violation_assignments_violation_id"),
|
|
"violation_assignments",
|
|
["violation_id"],
|
|
unique=False,
|
|
)
|
|
|
|
# Create violation_comments table
|
|
op.create_table(
|
|
"violation_comments",
|
|
sa.Column("id", sa.Integer(), nullable=False),
|
|
sa.Column("violation_id", sa.Integer(), nullable=False),
|
|
sa.Column("user_id", sa.Integer(), nullable=False),
|
|
sa.Column("comment", sa.Text(), nullable=False),
|
|
sa.Column(
|
|
"created_at",
|
|
sa.DateTime(timezone=True),
|
|
server_default=sa.text("CURRENT_TIMESTAMP"),
|
|
nullable=False,
|
|
),
|
|
sa.ForeignKeyConstraint(
|
|
["user_id"],
|
|
["users.id"],
|
|
),
|
|
sa.ForeignKeyConstraint(
|
|
["violation_id"],
|
|
["architecture_violations.id"],
|
|
),
|
|
sa.PrimaryKeyConstraint("id"),
|
|
)
|
|
op.create_index(
|
|
op.f("ix_violation_comments_id"), "violation_comments", ["id"], unique=False
|
|
)
|
|
op.create_index(
|
|
op.f("ix_violation_comments_violation_id"),
|
|
"violation_comments",
|
|
["violation_id"],
|
|
unique=False,
|
|
)
|
|
|
|
|
|
def downgrade() -> None:
|
|
# Drop tables in reverse order (to respect foreign key constraints)
|
|
op.drop_index(
|
|
op.f("ix_violation_comments_violation_id"), table_name="violation_comments"
|
|
)
|
|
op.drop_index(op.f("ix_violation_comments_id"), table_name="violation_comments")
|
|
op.drop_table("violation_comments")
|
|
|
|
op.drop_index(
|
|
op.f("ix_violation_assignments_violation_id"),
|
|
table_name="violation_assignments",
|
|
)
|
|
op.drop_index(
|
|
op.f("ix_violation_assignments_id"), table_name="violation_assignments"
|
|
)
|
|
op.drop_table("violation_assignments")
|
|
|
|
op.drop_index(
|
|
op.f("ix_architecture_violations_status"), table_name="architecture_violations"
|
|
)
|
|
op.drop_index(
|
|
op.f("ix_architecture_violations_severity"),
|
|
table_name="architecture_violations",
|
|
)
|
|
op.drop_index(
|
|
op.f("ix_architecture_violations_scan_id"), table_name="architecture_violations"
|
|
)
|
|
op.drop_index(
|
|
op.f("ix_architecture_violations_rule_id"), table_name="architecture_violations"
|
|
)
|
|
op.drop_index(
|
|
op.f("ix_architecture_violations_id"), table_name="architecture_violations"
|
|
)
|
|
op.drop_index(
|
|
op.f("ix_architecture_violations_file_path"),
|
|
table_name="architecture_violations",
|
|
)
|
|
op.drop_table("architecture_violations")
|
|
|
|
op.drop_index(
|
|
op.f("ix_architecture_rules_rule_id"), table_name="architecture_rules"
|
|
)
|
|
op.drop_index(op.f("ix_architecture_rules_id"), table_name="architecture_rules")
|
|
op.drop_table("architecture_rules")
|
|
|
|
op.drop_index(
|
|
op.f("ix_architecture_scans_timestamp"), table_name="architecture_scans"
|
|
)
|
|
op.drop_index(op.f("ix_architecture_scans_id"), table_name="architecture_scans")
|
|
op.drop_table("architecture_scans")
|