2026-02-14 15:26:42 +01:00
|
|
|
-- Universal Apply Page: Make Project.roundId nullable and add programId FK
|
|
|
|
|
-- This migration enables projects to be submitted to a program/edition without being assigned to a specific round
|
|
|
|
|
-- NOTE: Written to be idempotent (safe to re-run if partially applied)
|
|
|
|
|
|
|
|
|
|
-- Step 1: Add Program.slug for edition-wide apply URLs (nullable for existing programs)
|
|
|
|
|
ALTER TABLE "Program" ADD COLUMN IF NOT EXISTS "slug" TEXT;
|
|
|
|
|
CREATE UNIQUE INDEX IF NOT EXISTS "Program_slug_key" ON "Program"("slug");
|
|
|
|
|
|
|
|
|
|
-- Step 2: Add programId column (nullable initially to handle existing data)
|
|
|
|
|
ALTER TABLE "Project" ADD COLUMN IF NOT EXISTS "programId" TEXT;
|
|
|
|
|
|
|
|
|
|
-- Step 3: Backfill programId from existing round relationships
|
|
|
|
|
-- Only update rows where programId is still NULL (idempotent)
|
|
|
|
|
UPDATE "Project" p
|
|
|
|
|
SET "programId" = r."programId"
|
|
|
|
|
FROM "Round" r
|
|
|
|
|
WHERE p."roundId" = r.id
|
|
|
|
|
AND p."programId" IS NULL;
|
|
|
|
|
|
|
|
|
|
-- Step 4: Handle orphaned projects (no roundId = no way to derive programId)
|
|
|
|
|
-- Assign them to the first available program, or delete them if no program exists
|
|
|
|
|
DO $$
|
|
|
|
|
DECLARE
|
|
|
|
|
null_count INTEGER;
|
|
|
|
|
fallback_program_id TEXT;
|
|
|
|
|
BEGIN
|
|
|
|
|
SELECT COUNT(*) INTO null_count FROM "Project" WHERE "programId" IS NULL;
|
|
|
|
|
IF null_count > 0 THEN
|
|
|
|
|
SELECT id INTO fallback_program_id FROM "Program" ORDER BY "createdAt" ASC LIMIT 1;
|
|
|
|
|
IF fallback_program_id IS NOT NULL THEN
|
|
|
|
|
UPDATE "Project" SET "programId" = fallback_program_id WHERE "programId" IS NULL;
|
|
|
|
|
RAISE NOTICE 'Assigned % orphaned projects to fallback program %', null_count, fallback_program_id;
|
|
|
|
|
ELSE
|
|
|
|
|
DELETE FROM "Project" WHERE "programId" IS NULL;
|
|
|
|
|
RAISE NOTICE 'Deleted % orphaned projects (no program exists to assign them to)', null_count;
|
|
|
|
|
END IF;
|
|
|
|
|
END IF;
|
|
|
|
|
END $$;
|
|
|
|
|
|
|
|
|
|
-- Step 5: Make programId required (NOT NULL constraint) - safe if already NOT NULL
|
|
|
|
|
DO $$
|
|
|
|
|
BEGIN
|
|
|
|
|
IF EXISTS (
|
|
|
|
|
SELECT 1 FROM information_schema.columns
|
|
|
|
|
WHERE table_name = 'Project' AND column_name = 'programId' AND is_nullable = 'YES'
|
|
|
|
|
) THEN
|
|
|
|
|
ALTER TABLE "Project" ALTER COLUMN "programId" SET NOT NULL;
|
|
|
|
|
END IF;
|
|
|
|
|
END $$;
|
|
|
|
|
|
|
|
|
|
-- Step 6: Add foreign key constraint for programId (skip if already exists)
|
|
|
|
|
DO $$
|
|
|
|
|
BEGIN
|
|
|
|
|
IF NOT EXISTS (
|
|
|
|
|
SELECT 1 FROM information_schema.table_constraints
|
|
|
|
|
WHERE constraint_name = 'Project_programId_fkey' AND table_name = 'Project'
|
|
|
|
|
) THEN
|
|
|
|
|
ALTER TABLE "Project" ADD CONSTRAINT "Project_programId_fkey"
|
|
|
|
|
FOREIGN KEY ("programId") REFERENCES "Program"("id") ON DELETE CASCADE;
|
|
|
|
|
END IF;
|
|
|
|
|
END $$;
|
|
|
|
|
|
|
|
|
|
-- Step 7: Make roundId nullable (allow projects without round assignment)
|
|
|
|
|
-- Safe: DROP NOT NULL is idempotent if already nullable
|
|
|
|
|
DO $$
|
|
|
|
|
BEGIN
|
|
|
|
|
IF EXISTS (
|
|
|
|
|
SELECT 1 FROM information_schema.columns
|
|
|
|
|
WHERE table_name = 'Project' AND column_name = 'roundId' AND is_nullable = 'NO'
|
|
|
|
|
) THEN
|
|
|
|
|
ALTER TABLE "Project" ALTER COLUMN "roundId" DROP NOT NULL;
|
|
|
|
|
END IF;
|
|
|
|
|
END $$;
|
|
|
|
|
|
|
|
|
|
-- Step 8: Update round FK to SET NULL on delete (instead of CASCADE)
|
|
|
|
|
-- Projects should remain in the database if their round is deleted
|
|
|
|
|
DO $$
|
|
|
|
|
BEGIN
|
|
|
|
|
IF EXISTS (
|
|
|
|
|
SELECT 1 FROM information_schema.table_constraints
|
|
|
|
|
WHERE constraint_name = 'Project_roundId_fkey' AND table_name = 'Project'
|
|
|
|
|
) THEN
|
|
|
|
|
ALTER TABLE "Project" DROP CONSTRAINT "Project_roundId_fkey";
|
|
|
|
|
END IF;
|
|
|
|
|
ALTER TABLE "Project" ADD CONSTRAINT "Project_roundId_fkey"
|
|
|
|
|
FOREIGN KEY ("roundId") REFERENCES "Round"("id") ON DELETE SET NULL;
|
|
|
|
|
END $$;
|
|
|
|
|
|
|
|
|
|
-- Step 9: Add performance indexes
|
|
|
|
|
CREATE INDEX IF NOT EXISTS "Project_programId_idx" ON "Project"("programId");
|
|
|
|
|
CREATE INDEX IF NOT EXISTS "Project_programId_roundId_idx" ON "Project"("programId", "roundId");
|