2026-03-16 17:03:23 -05:00
|
|
|
import { createHash } from "node:crypto";
|
|
|
|
|
import fs from "node:fs";
|
|
|
|
|
import { afterEach, describe, expect, it } from "vitest";
|
|
|
|
|
import postgres from "postgres";
|
|
|
|
|
import {
|
|
|
|
|
applyPendingMigrations,
|
|
|
|
|
inspectMigrations,
|
|
|
|
|
} from "./client.js";
|
2026-03-26 11:04:07 -05:00
|
|
|
import {
|
|
|
|
|
getEmbeddedPostgresTestSupport,
|
|
|
|
|
startEmbeddedPostgresTestDatabase,
|
|
|
|
|
} from "./test-embedded-postgres.js";
|
2026-03-16 17:03:23 -05:00
|
|
|
|
2026-03-26 11:04:07 -05:00
|
|
|
const cleanups: Array<() => Promise<void>> = [];
|
|
|
|
|
const embeddedPostgresSupport = await getEmbeddedPostgresTestSupport();
|
|
|
|
|
const describeEmbeddedPostgres = embeddedPostgresSupport.supported ? describe : describe.skip;
|
2026-03-16 17:03:23 -05:00
|
|
|
|
|
|
|
|
async function createTempDatabase(): Promise<string> {
|
2026-03-26 11:04:07 -05:00
|
|
|
const db = await startEmbeddedPostgresTestDatabase("paperclip-db-client-");
|
|
|
|
|
cleanups.push(db.cleanup);
|
|
|
|
|
return db.connectionString;
|
2026-03-16 17:03:23 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function migrationHash(migrationFile: string): Promise<string> {
|
|
|
|
|
const content = await fs.promises.readFile(
|
|
|
|
|
new URL(`./migrations/${migrationFile}`, import.meta.url),
|
|
|
|
|
"utf8",
|
|
|
|
|
);
|
|
|
|
|
return createHash("sha256").update(content).digest("hex");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
afterEach(async () => {
|
2026-03-26 11:04:07 -05:00
|
|
|
while (cleanups.length > 0) {
|
|
|
|
|
const cleanup = cleanups.pop();
|
|
|
|
|
await cleanup?.();
|
2026-03-16 17:03:23 -05:00
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
2026-03-26 11:04:07 -05:00
|
|
|
if (!embeddedPostgresSupport.supported) {
|
|
|
|
|
console.warn(
|
|
|
|
|
`Skipping embedded Postgres migration tests on this host: ${embeddedPostgresSupport.reason ?? "unsupported environment"}`,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
describeEmbeddedPostgres("applyPendingMigrations", () => {
|
2026-03-16 17:03:23 -05:00
|
|
|
it(
|
|
|
|
|
"applies an inserted earlier migration without replaying later legacy migrations",
|
|
|
|
|
async () => {
|
|
|
|
|
const connectionString = await createTempDatabase();
|
|
|
|
|
|
|
|
|
|
await applyPendingMigrations(connectionString);
|
|
|
|
|
|
|
|
|
|
const sql = postgres(connectionString, { max: 1, onnotice: () => {} });
|
|
|
|
|
try {
|
|
|
|
|
const richMagnetoHash = await migrationHash("0030_rich_magneto.sql");
|
|
|
|
|
|
|
|
|
|
await sql.unsafe(
|
|
|
|
|
`DELETE FROM "drizzle"."__drizzle_migrations" WHERE hash = '${richMagnetoHash}'`,
|
|
|
|
|
);
|
|
|
|
|
await sql.unsafe(`DROP TABLE "company_logos"`);
|
|
|
|
|
} finally {
|
|
|
|
|
await sql.end();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const pendingState = await inspectMigrations(connectionString);
|
|
|
|
|
expect(pendingState).toMatchObject({
|
|
|
|
|
status: "needsMigrations",
|
|
|
|
|
pendingMigrations: ["0030_rich_magneto.sql"],
|
|
|
|
|
reason: "pending-migrations",
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
await applyPendingMigrations(connectionString);
|
|
|
|
|
|
|
|
|
|
const finalState = await inspectMigrations(connectionString);
|
|
|
|
|
expect(finalState.status).toBe("upToDate");
|
|
|
|
|
|
|
|
|
|
const verifySql = postgres(connectionString, { max: 1, onnotice: () => {} });
|
|
|
|
|
try {
|
|
|
|
|
const rows = await verifySql.unsafe<{ table_name: string }[]>(
|
|
|
|
|
`
|
|
|
|
|
SELECT table_name
|
|
|
|
|
FROM information_schema.tables
|
|
|
|
|
WHERE table_schema = 'public'
|
|
|
|
|
AND table_name IN ('company_logos', 'execution_workspaces')
|
|
|
|
|
ORDER BY table_name
|
|
|
|
|
`,
|
|
|
|
|
);
|
|
|
|
|
expect(rows.map((row) => row.table_name)).toEqual([
|
|
|
|
|
"company_logos",
|
|
|
|
|
"execution_workspaces",
|
|
|
|
|
]);
|
|
|
|
|
} finally {
|
|
|
|
|
await verifySql.end();
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
20_000,
|
|
|
|
|
);
|
2026-03-23 08:08:03 -05:00
|
|
|
|
|
|
|
|
it(
|
|
|
|
|
"replays migration 0044 safely when its schema changes already exist",
|
|
|
|
|
async () => {
|
|
|
|
|
const connectionString = await createTempDatabase();
|
|
|
|
|
|
|
|
|
|
await applyPendingMigrations(connectionString);
|
|
|
|
|
|
|
|
|
|
const sql = postgres(connectionString, { max: 1, onnotice: () => {} });
|
|
|
|
|
try {
|
|
|
|
|
const illegalToadHash = await migrationHash("0044_illegal_toad.sql");
|
|
|
|
|
|
|
|
|
|
await sql.unsafe(
|
|
|
|
|
`DELETE FROM "drizzle"."__drizzle_migrations" WHERE hash = '${illegalToadHash}'`,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const columns = await sql.unsafe<{ column_name: string }[]>(
|
|
|
|
|
`
|
|
|
|
|
SELECT column_name
|
|
|
|
|
FROM information_schema.columns
|
|
|
|
|
WHERE table_schema = 'public'
|
|
|
|
|
AND table_name = 'instance_settings'
|
|
|
|
|
AND column_name = 'general'
|
|
|
|
|
`,
|
|
|
|
|
);
|
|
|
|
|
expect(columns).toHaveLength(1);
|
|
|
|
|
} finally {
|
|
|
|
|
await sql.end();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const pendingState = await inspectMigrations(connectionString);
|
|
|
|
|
expect(pendingState).toMatchObject({
|
|
|
|
|
status: "needsMigrations",
|
|
|
|
|
pendingMigrations: ["0044_illegal_toad.sql"],
|
|
|
|
|
reason: "pending-migrations",
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
await applyPendingMigrations(connectionString);
|
|
|
|
|
|
|
|
|
|
const finalState = await inspectMigrations(connectionString);
|
|
|
|
|
expect(finalState.status).toBe("upToDate");
|
|
|
|
|
},
|
|
|
|
|
20_000,
|
|
|
|
|
);
|
2026-03-23 08:45:56 -05:00
|
|
|
|
|
|
|
|
it(
|
|
|
|
|
"enforces a unique board_api_keys.key_hash after migration 0044",
|
|
|
|
|
async () => {
|
|
|
|
|
const connectionString = await createTempDatabase();
|
|
|
|
|
|
|
|
|
|
await applyPendingMigrations(connectionString);
|
|
|
|
|
|
|
|
|
|
const sql = postgres(connectionString, { max: 1, onnotice: () => {} });
|
|
|
|
|
try {
|
|
|
|
|
await sql.unsafe(`
|
|
|
|
|
INSERT INTO "user" ("id", "name", "email", "email_verified", "created_at", "updated_at")
|
|
|
|
|
VALUES ('user-1', 'User One', 'user@example.com', true, now(), now())
|
|
|
|
|
`);
|
|
|
|
|
await sql.unsafe(`
|
|
|
|
|
INSERT INTO "board_api_keys" ("id", "user_id", "name", "key_hash", "created_at")
|
|
|
|
|
VALUES ('00000000-0000-0000-0000-000000000001', 'user-1', 'Key One', 'dup-hash', now())
|
|
|
|
|
`);
|
|
|
|
|
await expect(
|
|
|
|
|
sql.unsafe(`
|
|
|
|
|
INSERT INTO "board_api_keys" ("id", "user_id", "name", "key_hash", "created_at")
|
|
|
|
|
VALUES ('00000000-0000-0000-0000-000000000002', 'user-1', 'Key Two', 'dup-hash', now())
|
|
|
|
|
`),
|
|
|
|
|
).rejects.toThrow();
|
|
|
|
|
} finally {
|
|
|
|
|
await sql.end();
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
20_000,
|
|
|
|
|
);
|
2026-03-31 08:09:00 -05:00
|
|
|
|
|
|
|
|
it(
|
|
|
|
|
"replays migration 0046 safely when document revision columns already exist",
|
|
|
|
|
async () => {
|
|
|
|
|
const connectionString = await createTempDatabase();
|
|
|
|
|
|
|
|
|
|
await applyPendingMigrations(connectionString);
|
|
|
|
|
|
|
|
|
|
const sql = postgres(connectionString, { max: 1, onnotice: () => {} });
|
|
|
|
|
try {
|
|
|
|
|
const smoothSentinelsHash = await migrationHash("0046_smooth_sentinels.sql");
|
|
|
|
|
|
|
|
|
|
await sql.unsafe(
|
|
|
|
|
`DELETE FROM "drizzle"."__drizzle_migrations" WHERE hash = '${smoothSentinelsHash}'`,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const columns = await sql.unsafe<{ column_name: string; is_nullable: string; column_default: string | null }[]>(
|
|
|
|
|
`
|
|
|
|
|
SELECT column_name, is_nullable, column_default
|
|
|
|
|
FROM information_schema.columns
|
|
|
|
|
WHERE table_schema = 'public'
|
|
|
|
|
AND table_name = 'document_revisions'
|
|
|
|
|
AND column_name IN ('title', 'format')
|
|
|
|
|
ORDER BY column_name
|
|
|
|
|
`,
|
|
|
|
|
);
|
|
|
|
|
expect(columns).toHaveLength(2);
|
|
|
|
|
} finally {
|
|
|
|
|
await sql.end();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const pendingState = await inspectMigrations(connectionString);
|
|
|
|
|
expect(pendingState).toMatchObject({
|
|
|
|
|
status: "needsMigrations",
|
|
|
|
|
pendingMigrations: ["0046_smooth_sentinels.sql"],
|
|
|
|
|
reason: "pending-migrations",
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
await applyPendingMigrations(connectionString);
|
|
|
|
|
|
|
|
|
|
const finalState = await inspectMigrations(connectionString);
|
|
|
|
|
expect(finalState.status).toBe("upToDate");
|
|
|
|
|
|
|
|
|
|
const verifySql = postgres(connectionString, { max: 1, onnotice: () => {} });
|
|
|
|
|
try {
|
|
|
|
|
const columns = await verifySql.unsafe<{ column_name: string; is_nullable: string; column_default: string | null }[]>(
|
|
|
|
|
`
|
|
|
|
|
SELECT column_name, is_nullable, column_default
|
|
|
|
|
FROM information_schema.columns
|
|
|
|
|
WHERE table_schema = 'public'
|
|
|
|
|
AND table_name = 'document_revisions'
|
|
|
|
|
AND column_name IN ('title', 'format')
|
|
|
|
|
ORDER BY column_name
|
|
|
|
|
`,
|
|
|
|
|
);
|
|
|
|
|
expect(columns).toEqual([
|
|
|
|
|
expect.objectContaining({
|
|
|
|
|
column_name: "format",
|
|
|
|
|
is_nullable: "NO",
|
|
|
|
|
}),
|
|
|
|
|
expect.objectContaining({
|
|
|
|
|
column_name: "title",
|
|
|
|
|
is_nullable: "YES",
|
|
|
|
|
}),
|
|
|
|
|
]);
|
|
|
|
|
expect(columns[0]?.column_default).toContain("'markdown'");
|
|
|
|
|
} finally {
|
|
|
|
|
await verifySql.end();
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
20_000,
|
|
|
|
|
);
|
2026-04-02 10:54:56 -05:00
|
|
|
|
|
|
|
|
it(
|
|
|
|
|
"replays migration 0047 safely when feedback tables and run columns already exist",
|
|
|
|
|
async () => {
|
|
|
|
|
const connectionString = await createTempDatabase();
|
|
|
|
|
|
|
|
|
|
await applyPendingMigrations(connectionString);
|
|
|
|
|
|
|
|
|
|
const sql = postgres(connectionString, { max: 1, onnotice: () => {} });
|
|
|
|
|
try {
|
|
|
|
|
const overjoyedGrootHash = await migrationHash("0047_overjoyed_groot.sql");
|
|
|
|
|
|
|
|
|
|
await sql.unsafe(
|
|
|
|
|
`DELETE FROM "drizzle"."__drizzle_migrations" WHERE hash = '${overjoyedGrootHash}'`,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const tables = await sql.unsafe<{ table_name: string }[]>(
|
|
|
|
|
`
|
|
|
|
|
SELECT table_name
|
|
|
|
|
FROM information_schema.tables
|
|
|
|
|
WHERE table_schema = 'public'
|
|
|
|
|
AND table_name IN ('feedback_exports', 'feedback_votes')
|
|
|
|
|
ORDER BY table_name
|
|
|
|
|
`,
|
|
|
|
|
);
|
|
|
|
|
expect(tables.map((row) => row.table_name)).toEqual([
|
|
|
|
|
"feedback_exports",
|
|
|
|
|
"feedback_votes",
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
const columns = await sql.unsafe<{ table_name: string; column_name: string }[]>(
|
|
|
|
|
`
|
|
|
|
|
SELECT table_name, column_name
|
|
|
|
|
FROM information_schema.columns
|
|
|
|
|
WHERE table_schema = 'public'
|
|
|
|
|
AND (
|
|
|
|
|
(table_name = 'companies' AND column_name IN (
|
|
|
|
|
'feedback_data_sharing_enabled',
|
|
|
|
|
'feedback_data_sharing_consent_at',
|
|
|
|
|
'feedback_data_sharing_consent_by_user_id',
|
|
|
|
|
'feedback_data_sharing_terms_version'
|
|
|
|
|
))
|
|
|
|
|
OR (table_name = 'document_revisions' AND column_name = 'created_by_run_id')
|
|
|
|
|
OR (table_name = 'issue_comments' AND column_name = 'created_by_run_id')
|
|
|
|
|
)
|
|
|
|
|
ORDER BY table_name, column_name
|
|
|
|
|
`,
|
|
|
|
|
);
|
|
|
|
|
expect(columns).toHaveLength(6);
|
|
|
|
|
} finally {
|
|
|
|
|
await sql.end();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const pendingState = await inspectMigrations(connectionString);
|
|
|
|
|
expect(pendingState).toMatchObject({
|
|
|
|
|
status: "needsMigrations",
|
|
|
|
|
pendingMigrations: ["0047_overjoyed_groot.sql"],
|
|
|
|
|
reason: "pending-migrations",
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
await applyPendingMigrations(connectionString);
|
|
|
|
|
|
|
|
|
|
const finalState = await inspectMigrations(connectionString);
|
|
|
|
|
expect(finalState.status).toBe("upToDate");
|
2026-04-02 11:38:57 -05:00
|
|
|
|
|
|
|
|
const verifySql = postgres(connectionString, { max: 1, onnotice: () => {} });
|
|
|
|
|
try {
|
|
|
|
|
const constraints = await verifySql.unsafe<{ conname: string }[]>(
|
|
|
|
|
`
|
|
|
|
|
SELECT conname
|
|
|
|
|
FROM pg_constraint
|
|
|
|
|
WHERE conname IN (
|
|
|
|
|
'feedback_exports_company_id_companies_id_fk',
|
|
|
|
|
'feedback_exports_feedback_vote_id_feedback_votes_id_fk',
|
|
|
|
|
'feedback_exports_issue_id_issues_id_fk',
|
|
|
|
|
'feedback_votes_company_id_companies_id_fk',
|
|
|
|
|
'feedback_votes_issue_id_issues_id_fk'
|
|
|
|
|
)
|
|
|
|
|
ORDER BY conname
|
|
|
|
|
`,
|
|
|
|
|
);
|
|
|
|
|
expect(constraints.map((row) => row.conname)).toEqual([
|
|
|
|
|
"feedback_exports_company_id_companies_id_fk",
|
|
|
|
|
"feedback_exports_feedback_vote_id_feedback_votes_id_fk",
|
|
|
|
|
"feedback_exports_issue_id_issues_id_fk",
|
|
|
|
|
"feedback_votes_company_id_companies_id_fk",
|
|
|
|
|
"feedback_votes_issue_id_issues_id_fk",
|
|
|
|
|
]);
|
|
|
|
|
} finally {
|
|
|
|
|
await verifySql.end();
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
20_000,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
it(
|
|
|
|
|
"replays migration 0048 safely when routines.variables already exists",
|
|
|
|
|
async () => {
|
|
|
|
|
const connectionString = await createTempDatabase();
|
|
|
|
|
|
|
|
|
|
await applyPendingMigrations(connectionString);
|
|
|
|
|
|
|
|
|
|
const sql = postgres(connectionString, { max: 1, onnotice: () => {} });
|
|
|
|
|
try {
|
|
|
|
|
const flashyMarrowHash = await migrationHash("0048_flashy_marrow.sql");
|
|
|
|
|
|
|
|
|
|
await sql.unsafe(
|
|
|
|
|
`DELETE FROM "drizzle"."__drizzle_migrations" WHERE hash = '${flashyMarrowHash}'`,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const columns = await sql.unsafe<{ column_name: string }[]>(
|
|
|
|
|
`
|
|
|
|
|
SELECT column_name
|
|
|
|
|
FROM information_schema.columns
|
|
|
|
|
WHERE table_schema = 'public'
|
|
|
|
|
AND table_name = 'routines'
|
|
|
|
|
AND column_name = 'variables'
|
|
|
|
|
`,
|
|
|
|
|
);
|
|
|
|
|
expect(columns).toHaveLength(1);
|
|
|
|
|
} finally {
|
|
|
|
|
await sql.end();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const pendingState = await inspectMigrations(connectionString);
|
|
|
|
|
expect(pendingState).toMatchObject({
|
|
|
|
|
status: "needsMigrations",
|
|
|
|
|
pendingMigrations: ["0048_flashy_marrow.sql"],
|
|
|
|
|
reason: "pending-migrations",
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
await applyPendingMigrations(connectionString);
|
|
|
|
|
|
|
|
|
|
const finalState = await inspectMigrations(connectionString);
|
|
|
|
|
expect(finalState.status).toBe("upToDate");
|
|
|
|
|
|
|
|
|
|
const verifySql = postgres(connectionString, { max: 1, onnotice: () => {} });
|
|
|
|
|
try {
|
|
|
|
|
const columns = await verifySql.unsafe<{ column_name: string; is_nullable: string; data_type: string }[]>(
|
|
|
|
|
`
|
|
|
|
|
SELECT column_name, is_nullable, data_type
|
|
|
|
|
FROM information_schema.columns
|
|
|
|
|
WHERE table_schema = 'public'
|
|
|
|
|
AND table_name = 'routines'
|
|
|
|
|
AND column_name = 'variables'
|
|
|
|
|
`,
|
|
|
|
|
);
|
|
|
|
|
expect(columns).toEqual([
|
|
|
|
|
expect.objectContaining({
|
|
|
|
|
column_name: "variables",
|
|
|
|
|
is_nullable: "NO",
|
|
|
|
|
data_type: "jsonb",
|
|
|
|
|
}),
|
|
|
|
|
]);
|
|
|
|
|
} finally {
|
|
|
|
|
await verifySql.end();
|
|
|
|
|
}
|
2026-04-02 10:54:56 -05:00
|
|
|
},
|
|
|
|
|
20_000,
|
|
|
|
|
);
|
2026-04-06 10:05:41 -05:00
|
|
|
|
|
|
|
|
it(
|
|
|
|
|
"replays migration 0050 safely when projects.env already exists",
|
|
|
|
|
async () => {
|
|
|
|
|
const connectionString = await createTempDatabase();
|
|
|
|
|
|
|
|
|
|
await applyPendingMigrations(connectionString);
|
|
|
|
|
|
|
|
|
|
const sql = postgres(connectionString, { max: 1, onnotice: () => {} });
|
|
|
|
|
try {
|
|
|
|
|
const stiffLuckmanHash = await migrationHash("0050_stiff_luckman.sql");
|
|
|
|
|
|
|
|
|
|
await sql.unsafe(
|
|
|
|
|
`DELETE FROM "drizzle"."__drizzle_migrations" WHERE hash = '${stiffLuckmanHash}'`,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const columns = await sql.unsafe<{ column_name: string }[]>(
|
|
|
|
|
`
|
|
|
|
|
SELECT column_name
|
|
|
|
|
FROM information_schema.columns
|
|
|
|
|
WHERE table_schema = 'public'
|
|
|
|
|
AND table_name = 'projects'
|
|
|
|
|
AND column_name = 'env'
|
|
|
|
|
`,
|
|
|
|
|
);
|
|
|
|
|
expect(columns).toHaveLength(1);
|
|
|
|
|
} finally {
|
|
|
|
|
await sql.end();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const pendingState = await inspectMigrations(connectionString);
|
|
|
|
|
expect(pendingState).toMatchObject({
|
|
|
|
|
status: "needsMigrations",
|
|
|
|
|
pendingMigrations: ["0050_stiff_luckman.sql"],
|
|
|
|
|
reason: "pending-migrations",
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
await applyPendingMigrations(connectionString);
|
|
|
|
|
|
|
|
|
|
const finalState = await inspectMigrations(connectionString);
|
|
|
|
|
expect(finalState.status).toBe("upToDate");
|
|
|
|
|
|
|
|
|
|
const verifySql = postgres(connectionString, { max: 1, onnotice: () => {} });
|
|
|
|
|
try {
|
|
|
|
|
const columns = await verifySql.unsafe<{ column_name: string; is_nullable: string; data_type: string }[]>(
|
|
|
|
|
`
|
|
|
|
|
SELECT column_name, is_nullable, data_type
|
|
|
|
|
FROM information_schema.columns
|
|
|
|
|
WHERE table_schema = 'public'
|
|
|
|
|
AND table_name = 'projects'
|
|
|
|
|
AND column_name = 'env'
|
|
|
|
|
`,
|
|
|
|
|
);
|
|
|
|
|
expect(columns).toEqual([
|
|
|
|
|
expect.objectContaining({
|
|
|
|
|
column_name: "env",
|
|
|
|
|
is_nullable: "YES",
|
|
|
|
|
data_type: "jsonb",
|
|
|
|
|
}),
|
|
|
|
|
]);
|
|
|
|
|
} finally {
|
|
|
|
|
await verifySql.end();
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
20_000,
|
|
|
|
|
);
|
[codex] Add plugin orchestration host APIs (#4114)
## Thinking Path
> - Paperclip orchestrates AI agents for zero-human companies.
> - The plugin system is the extension path for optional capabilities
that should not require core product changes for every integration.
> - Plugins need scoped host APIs for issue orchestration, documents,
wakeups, summaries, activity attribution, and isolated database state.
> - Without those host APIs, richer plugins either cannot coordinate
Paperclip work safely or need privileged core-side special cases.
> - This pull request adds the plugin orchestration host surface, scoped
route dispatch, a database namespace layer, and a smoke plugin that
exercises the contract.
> - The benefit is a broader plugin API that remains company-scoped,
auditable, and covered by tests.
## What Changed
- Added plugin orchestration host APIs for issue creation, document
access, wakeups, summaries, plugin-origin activity, and scoped API route
dispatch.
- Added plugin database namespace tables, schema exports, migration
checks, and idempotent replay coverage under migration
`0059_plugin_database_namespaces`.
- Added shared plugin route/API types and validators used by server and
SDK boundaries.
- Expanded plugin SDK types, protocol helpers, worker RPC host behavior,
and testing utilities for orchestration flows.
- Added the `plugin-orchestration-smoke-example` package to exercise
scoped routes, restricted database namespaces, issue orchestration,
documents, wakeups, summaries, and UI status surfaces.
- Kept the new orchestration smoke fixture out of the root pnpm
workspace importer so this PR preserves the repository policy of not
committing `pnpm-lock.yaml`.
- Updated plugin docs and database docs for the new orchestration and
database namespace surfaces.
- Rebased the branch onto `public-gh/master`, resolved conflicts, and
removed `pnpm-lock.yaml` from the final PR diff.
## Verification
- `pnpm install --frozen-lockfile`
- `pnpm --filter @paperclipai/db typecheck`
- `pnpm exec vitest run packages/db/src/client.test.ts`
- `pnpm exec vitest run server/src/__tests__/plugin-database.test.ts
server/src/__tests__/plugin-orchestration-apis.test.ts
server/src/__tests__/plugin-routes-authz.test.ts
server/src/__tests__/plugin-scoped-api-routes.test.ts
server/src/__tests__/plugin-sdk-orchestration-contract.test.ts`
- From `packages/plugins/examples/plugin-orchestration-smoke-example`:
`pnpm exec vitest run --config ./vitest.config.ts`
- `pnpm --dir
packages/plugins/examples/plugin-orchestration-smoke-example run
typecheck`
- `pnpm --filter @paperclipai/server typecheck`
- PR CI on latest head `293fc67c`: `policy`, `verify`, `e2e`, and
`security/snyk` all passed.
## Risks
- Medium risk: this expands plugin host authority, so route auth,
company scoping, and plugin-origin activity attribution need careful
review.
- Medium risk: database namespace migration behavior must remain
idempotent for environments that may have seen earlier branch versions.
- Medium risk: the orchestration smoke fixture is intentionally excluded
from the root workspace importer to avoid a `pnpm-lock.yaml` PR diff;
direct fixture verification remains listed above.
- Low operational risk from the PR setup itself: the branch is rebased
onto current `master`, the migration is ordered after upstream
`0057`/`0058`, and `pnpm-lock.yaml` is not in the final diff.
> For core feature work, check [`ROADMAP.md`](ROADMAP.md) first and
discuss it in `#dev` before opening the PR. Feature PRs that overlap
with planned core work may need to be redirected — check the roadmap
first. See `CONTRIBUTING.md`.
Roadmap checked: this work aligns with the completed Plugin system
milestone and extends the plugin surface rather than duplicating an
unrelated planned core feature.
## Model Used
- OpenAI Codex, GPT-5-based coding agent in a tool-enabled CLI
environment. Exact hosted model build and context-window size are not
exposed by the runtime; reasoning/tool use were enabled for repository
inspection, editing, testing, git operations, and PR creation.
## Checklist
- [x] I have included a thinking path that traces from project context
to this change
- [x] I have specified the model used (with version and capability
details)
- [x] I have checked ROADMAP.md and confirmed this PR does not duplicate
planned core work
- [x] I have run tests locally and they pass
- [x] I have added or updated tests where applicable
- [x] If this change affects the UI, I have included before/after
screenshots (N/A: no core UI screen change; example plugin UI contract
is covered by tests)
- [x] I have updated relevant documentation to reflect my changes
- [x] I have considered and documented any risks above
- [x] I will address all Greptile and reviewer comments before
requesting merge
---------
Co-authored-by: Paperclip <noreply@paperclip.ing>
2026-04-20 08:52:51 -05:00
|
|
|
|
|
|
|
|
it(
|
|
|
|
|
"replays migration 0059 safely when plugin_database_namespaces already exists",
|
|
|
|
|
async () => {
|
|
|
|
|
const connectionString = await createTempDatabase();
|
|
|
|
|
|
|
|
|
|
await applyPendingMigrations(connectionString);
|
|
|
|
|
|
|
|
|
|
const sql = postgres(connectionString, { max: 1, onnotice: () => {} });
|
|
|
|
|
try {
|
|
|
|
|
const pluginNamespacesHash = await migrationHash(
|
|
|
|
|
"0059_plugin_database_namespaces.sql",
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
await sql.unsafe(
|
|
|
|
|
`DELETE FROM "drizzle"."__drizzle_migrations" WHERE hash = '${pluginNamespacesHash}'`,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const tables = await sql.unsafe<{ table_name: string }[]>(
|
|
|
|
|
`
|
|
|
|
|
SELECT table_name
|
|
|
|
|
FROM information_schema.tables
|
|
|
|
|
WHERE table_schema = 'public'
|
|
|
|
|
AND table_name IN ('plugin_database_namespaces', 'plugin_migrations')
|
|
|
|
|
ORDER BY table_name
|
|
|
|
|
`,
|
|
|
|
|
);
|
|
|
|
|
expect(tables.map((row) => row.table_name)).toEqual([
|
|
|
|
|
"plugin_database_namespaces",
|
|
|
|
|
"plugin_migrations",
|
|
|
|
|
]);
|
|
|
|
|
} finally {
|
|
|
|
|
await sql.end();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const pendingState = await inspectMigrations(connectionString);
|
|
|
|
|
expect(pendingState).toMatchObject({
|
|
|
|
|
status: "needsMigrations",
|
|
|
|
|
pendingMigrations: ["0059_plugin_database_namespaces.sql"],
|
|
|
|
|
reason: "pending-migrations",
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
await applyPendingMigrations(connectionString);
|
|
|
|
|
|
|
|
|
|
const finalState = await inspectMigrations(connectionString);
|
|
|
|
|
expect(finalState.status).toBe("upToDate");
|
|
|
|
|
|
|
|
|
|
const verifySql = postgres(connectionString, { max: 1, onnotice: () => {} });
|
|
|
|
|
try {
|
|
|
|
|
const indexes = await verifySql.unsafe<{ indexname: string }[]>(
|
|
|
|
|
`
|
|
|
|
|
SELECT indexname
|
|
|
|
|
FROM pg_indexes
|
|
|
|
|
WHERE schemaname = 'public'
|
|
|
|
|
AND tablename IN ('plugin_database_namespaces', 'plugin_migrations')
|
|
|
|
|
ORDER BY indexname
|
|
|
|
|
`,
|
|
|
|
|
);
|
|
|
|
|
expect(indexes.map((row) => row.indexname)).toEqual(
|
|
|
|
|
expect.arrayContaining([
|
|
|
|
|
"plugin_database_namespaces_namespace_idx",
|
|
|
|
|
"plugin_database_namespaces_plugin_idx",
|
|
|
|
|
"plugin_database_namespaces_status_idx",
|
|
|
|
|
"plugin_migrations_plugin_idx",
|
|
|
|
|
"plugin_migrations_plugin_key_idx",
|
|
|
|
|
"plugin_migrations_status_idx",
|
|
|
|
|
]),
|
|
|
|
|
);
|
|
|
|
|
} finally {
|
|
|
|
|
await verifySql.end();
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
20_000,
|
|
|
|
|
);
|
2026-03-16 17:03:23 -05:00
|
|
|
});
|