|
| 1 | +-- DojoWatch Database Schema |
| 2 | +-- Run this against your Supabase project via the SQL editor or CLI. |
| 3 | + |
| 4 | +-- ─── Tables ────────────────────────────────────────────────────── |
| 5 | + |
| 6 | +-- One row per CI or local check execution |
| 7 | +CREATE TABLE IF NOT EXISTS vr_runs ( |
| 8 | + id UUID PRIMARY KEY DEFAULT gen_random_uuid(), |
| 9 | + project TEXT NOT NULL, |
| 10 | + pr_number INTEGER, |
| 11 | + branch TEXT NOT NULL, |
| 12 | + commit_sha TEXT NOT NULL, |
| 13 | + status TEXT NOT NULL DEFAULT 'pending' CHECK (status IN ('pending', 'pass', 'fail')), |
| 14 | + total_diffs INTEGER NOT NULL DEFAULT 0, |
| 15 | + regressions_count INTEGER NOT NULL DEFAULT 0, |
| 16 | + engine TEXT NOT NULL CHECK (engine IN ('claude', 'gemini')), |
| 17 | + created_at TIMESTAMPTZ NOT NULL DEFAULT now() |
| 18 | +); |
| 19 | + |
| 20 | +-- One row per changed screenshot in a run |
| 21 | +CREATE TABLE IF NOT EXISTS vr_diffs ( |
| 22 | + id UUID PRIMARY KEY DEFAULT gen_random_uuid(), |
| 23 | + run_id UUID NOT NULL REFERENCES vr_runs(id) ON DELETE CASCADE, |
| 24 | + name TEXT NOT NULL, |
| 25 | + viewport TEXT NOT NULL, |
| 26 | + baseline_storage_path TEXT, |
| 27 | + current_storage_path TEXT, |
| 28 | + diff_storage_path TEXT, |
| 29 | + pixel_diff_percent REAL NOT NULL DEFAULT 0, |
| 30 | + tier TEXT NOT NULL CHECK (tier IN ('SKIP', 'FAST_CHECK', 'FULL_ANALYSIS')), |
| 31 | + analysis JSONB, |
| 32 | + severity TEXT CHECK (severity IN ('high', 'medium', 'low')), |
| 33 | + review_status TEXT NOT NULL DEFAULT 'pending' CHECK (review_status IN ('pending', 'approved', 'rejected')), |
| 34 | + reviewed_by TEXT, |
| 35 | + reviewed_at TIMESTAMPTZ, |
| 36 | + created_at TIMESTAMPTZ NOT NULL DEFAULT now() |
| 37 | +); |
| 38 | + |
| 39 | +-- Source of truth for approved baseline screenshots |
| 40 | +CREATE TABLE IF NOT EXISTS vr_baselines ( |
| 41 | + id UUID PRIMARY KEY DEFAULT gen_random_uuid(), |
| 42 | + project TEXT NOT NULL, |
| 43 | + name TEXT NOT NULL, |
| 44 | + viewport TEXT NOT NULL, |
| 45 | + storage_path TEXT NOT NULL, |
| 46 | + hash TEXT NOT NULL, |
| 47 | + approved_at TIMESTAMPTZ NOT NULL DEFAULT now(), |
| 48 | + approved_by TEXT, |
| 49 | + UNIQUE (project, name, viewport) |
| 50 | +); |
| 51 | + |
| 52 | +-- ─── Indexes ───────────────────────────────────────────────────── |
| 53 | + |
| 54 | +CREATE INDEX IF NOT EXISTS idx_vr_runs_project ON vr_runs(project); |
| 55 | +CREATE INDEX IF NOT EXISTS idx_vr_runs_branch ON vr_runs(branch); |
| 56 | +CREATE INDEX IF NOT EXISTS idx_vr_runs_pr ON vr_runs(pr_number); |
| 57 | +CREATE INDEX IF NOT EXISTS idx_vr_diffs_run_id ON vr_diffs(run_id); |
| 58 | +CREATE INDEX IF NOT EXISTS idx_vr_diffs_review ON vr_diffs(review_status); |
| 59 | +CREATE INDEX IF NOT EXISTS idx_vr_baselines_project ON vr_baselines(project); |
| 60 | +CREATE INDEX IF NOT EXISTS idx_vr_baselines_lookup ON vr_baselines(project, name, viewport); |
| 61 | + |
| 62 | +-- ─── Row Level Security ────────────────────────────────────────── |
| 63 | + |
| 64 | +ALTER TABLE vr_runs ENABLE ROW LEVEL SECURITY; |
| 65 | +ALTER TABLE vr_diffs ENABLE ROW LEVEL SECURITY; |
| 66 | +ALTER TABLE vr_baselines ENABLE ROW LEVEL SECURITY; |
| 67 | + |
| 68 | +-- Service role (CI uploads) has full access |
| 69 | +CREATE POLICY "Service role full access on vr_runs" |
| 70 | + ON vr_runs FOR ALL |
| 71 | + USING (true) |
| 72 | + WITH CHECK (true); |
| 73 | + |
| 74 | +CREATE POLICY "Service role full access on vr_diffs" |
| 75 | + ON vr_diffs FOR ALL |
| 76 | + USING (true) |
| 77 | + WITH CHECK (true); |
| 78 | + |
| 79 | +CREATE POLICY "Service role full access on vr_baselines" |
| 80 | + ON vr_baselines FOR ALL |
| 81 | + USING (true) |
| 82 | + WITH CHECK (true); |
| 83 | + |
| 84 | +-- Anon key can read (for dashboard, when it's built) |
| 85 | +CREATE POLICY "Anon read on vr_runs" |
| 86 | + ON vr_runs FOR SELECT |
| 87 | + USING (true); |
| 88 | + |
| 89 | +CREATE POLICY "Anon read on vr_diffs" |
| 90 | + ON vr_diffs FOR SELECT |
| 91 | + USING (true); |
| 92 | + |
| 93 | +CREATE POLICY "Anon read on vr_baselines" |
| 94 | + ON vr_baselines FOR SELECT |
| 95 | + USING (true); |
| 96 | + |
| 97 | +-- ─── Storage Buckets ───────────────────────────────────────────── |
| 98 | +-- Create these via Supabase dashboard or CLI: |
| 99 | +-- supabase storage create baselines --public false |
| 100 | +-- supabase storage create captures --public false |
| 101 | +-- supabase storage create diffs --public false |
| 102 | +-- |
| 103 | +-- All buckets are private. Dashboard serves images via signed URLs. |
0 commit comments