@jagit/shared 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (54) hide show
  1. package/dist/approval-bridge.d.ts +12 -0
  2. package/dist/approval-bridge.js +35 -0
  3. package/dist/branch.d.ts +10 -0
  4. package/dist/branch.js +20 -0
  5. package/dist/branch.test.d.ts +1 -0
  6. package/dist/branch.test.js +32 -0
  7. package/dist/config.d.ts +19 -0
  8. package/dist/config.js +36 -0
  9. package/dist/config.test.d.ts +1 -0
  10. package/dist/config.test.js +31 -0
  11. package/dist/credentials.d.ts +82 -0
  12. package/dist/credentials.js +94 -0
  13. package/dist/credentials.test.d.ts +1 -0
  14. package/dist/credentials.test.js +206 -0
  15. package/dist/crypto.d.ts +12 -0
  16. package/dist/crypto.js +29 -0
  17. package/dist/crypto.test.d.ts +1 -0
  18. package/dist/crypto.test.js +20 -0
  19. package/dist/events.d.ts +13 -0
  20. package/dist/events.js +24 -0
  21. package/dist/git-worktree.d.ts +2 -0
  22. package/dist/git-worktree.js +14 -0
  23. package/dist/index.d.ts +15 -0
  24. package/dist/index.js +14 -0
  25. package/dist/mcp-config.d.ts +77 -0
  26. package/dist/mcp-config.js +71 -0
  27. package/dist/mcp-servers.d.ts +54 -0
  28. package/dist/mcp-servers.js +73 -0
  29. package/dist/prisma.d.ts +3 -0
  30. package/dist/prisma.js +20 -0
  31. package/dist/prisma.test.d.ts +1 -0
  32. package/dist/prisma.test.js +16 -0
  33. package/dist/queue.d.ts +10 -0
  34. package/dist/queue.js +7 -0
  35. package/dist/retry.d.ts +5 -0
  36. package/dist/retry.js +16 -0
  37. package/dist/retry.test.d.ts +1 -0
  38. package/dist/retry.test.js +24 -0
  39. package/dist/seed.d.ts +99 -0
  40. package/dist/seed.js +123 -0
  41. package/dist/seed.test.d.ts +1 -0
  42. package/dist/seed.test.js +126 -0
  43. package/dist/types.d.ts +16 -0
  44. package/dist/types.js +5 -0
  45. package/package.json +29 -0
  46. package/prisma/migrations/20260615000000_init/migration.sql +157 -0
  47. package/prisma/migrations/20260615000001_add_updated_at_agent_template/migration.sql +2 -0
  48. package/prisma/migrations/20260616000000_mcp_review/migration.sql +23 -0
  49. package/prisma/migrations/20260616100000_mcp_http_transport/migration.sql +5 -0
  50. package/prisma/migrations/20260616120000_review_default_false/migration.sql +1 -0
  51. package/prisma/migrations/20260620040203_add_usage_models/migration.sql +28 -0
  52. package/prisma/migrations/20260620120000_add_agent_session/migration.sql +35 -0
  53. package/prisma/migrations/migration_lock.toml +3 -0
  54. package/prisma/schema.prisma +257 -0
package/dist/seed.js ADDED
@@ -0,0 +1,123 @@
1
+ import { z } from "zod";
2
+ import { encrypt } from "./crypto.js";
3
+ const CredentialKindSchema = z.enum(["jira", "gitlab", "telegram", "anthropic"]);
4
+ const CredentialSeedSchema = z.object({
5
+ kind: CredentialKindSchema,
6
+ name: z.string().min(1),
7
+ secrets: z.record(z.string(), z.string().min(1)),
8
+ meta: z.record(z.string(), z.string()),
9
+ });
10
+ export const SeedDataSchema = z.object({
11
+ agentTemplate: z.object({
12
+ name: z.string().min(1),
13
+ model: z.string().min(1),
14
+ systemPrompt: z.string().min(1),
15
+ maxConcurrent: z.number().int().positive(),
16
+ allowedTools: z.array(z.string().min(1)),
17
+ skills: z.array(z.string()),
18
+ }),
19
+ credentials: z.array(CredentialSeedSchema).min(1),
20
+ repoMapping: z.object({
21
+ jiraProjectKey: z.string().min(1),
22
+ gitlabProjectId: z.string().min(1),
23
+ defaultBaseBranch: z.string().min(1),
24
+ branchPrefixRules: z.record(z.string(), z.string().min(1)),
25
+ agentTemplateName: z.string().min(1),
26
+ }),
27
+ });
28
+ export function buildSeedData(input) {
29
+ return SeedDataSchema.parse({
30
+ agentTemplate: {
31
+ name: "default",
32
+ model: "claude-opus-4-5",
33
+ systemPrompt: "You are JaGit's default coding agent. Implement assigned Jira issues, verify changes, and report concise progress.",
34
+ maxConcurrent: 1,
35
+ allowedTools: ["read_file", "write_file", "bash", "search"],
36
+ skills: [],
37
+ },
38
+ credentials: [
39
+ {
40
+ kind: "jira",
41
+ name: "default",
42
+ secrets: { email: "bot@example.com", token: "REPLACE_ME" },
43
+ meta: {
44
+ baseUrl: "https://your-org.atlassian.net",
45
+ botAccountId: "REPLACE_ME",
46
+ },
47
+ },
48
+ {
49
+ kind: "gitlab",
50
+ name: "default",
51
+ secrets: { token: "glpat-REPLACE_ME" },
52
+ meta: { baseUrl: "https://gitlab.com" },
53
+ },
54
+ {
55
+ kind: "telegram",
56
+ name: "default",
57
+ secrets: { botToken: "REPLACE_ME" },
58
+ meta: { chatId: "REPLACE_ME" },
59
+ },
60
+ {
61
+ kind: "anthropic",
62
+ name: "default",
63
+ secrets: { apiKey: input.anthropicApiKey },
64
+ meta: {},
65
+ },
66
+ ],
67
+ repoMapping: {
68
+ jiraProjectKey: "JAGIT",
69
+ gitlabProjectId: "your-namespace/your-repo",
70
+ defaultBaseBranch: "main",
71
+ branchPrefixRules: {
72
+ Bug: "bugfix/",
73
+ Story: "feature/",
74
+ Task: "feature/",
75
+ default: "feature/",
76
+ },
77
+ agentTemplateName: "default",
78
+ },
79
+ });
80
+ }
81
+ function encryptedSecrets(secrets, encryptionKey) {
82
+ return { encrypted: encrypt(JSON.stringify(secrets), encryptionKey) };
83
+ }
84
+ export async function seedDatabase(client, seedData, encryptionKey) {
85
+ const { name, ...agentTemplateUpdate } = seedData.agentTemplate;
86
+ const agentTemplate = await client.agentTemplate.upsert({
87
+ where: { name },
88
+ create: seedData.agentTemplate,
89
+ update: agentTemplateUpdate,
90
+ });
91
+ for (const credential of seedData.credentials) {
92
+ const secrets = encryptedSecrets(credential.secrets, encryptionKey);
93
+ await client.credential.upsert({
94
+ where: { kind_name: { kind: credential.kind, name: credential.name } },
95
+ create: {
96
+ kind: credential.kind,
97
+ name: credential.name,
98
+ secrets,
99
+ meta: credential.meta,
100
+ },
101
+ update: {
102
+ secrets,
103
+ meta: credential.meta,
104
+ },
105
+ });
106
+ }
107
+ await client.repoMapping.upsert({
108
+ where: { jiraProjectKey: seedData.repoMapping.jiraProjectKey },
109
+ create: {
110
+ jiraProjectKey: seedData.repoMapping.jiraProjectKey,
111
+ gitlabProjectId: seedData.repoMapping.gitlabProjectId,
112
+ defaultBaseBranch: seedData.repoMapping.defaultBaseBranch,
113
+ branchPrefixRules: seedData.repoMapping.branchPrefixRules,
114
+ agentTemplateId: agentTemplate.id,
115
+ },
116
+ update: {
117
+ gitlabProjectId: seedData.repoMapping.gitlabProjectId,
118
+ defaultBaseBranch: seedData.repoMapping.defaultBaseBranch,
119
+ branchPrefixRules: seedData.repoMapping.branchPrefixRules,
120
+ agentTemplateId: agentTemplate.id,
121
+ },
122
+ });
123
+ }
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,126 @@
1
+ import { describe, expect, it } from "vitest";
2
+ import { decrypt } from "./crypto.js";
3
+ import { buildSeedData, seedDatabase, SeedDataSchema, } from "./seed.js";
4
+ const KEY = Buffer.alloc(32, 11).toString("base64");
5
+ function createFakePrisma() {
6
+ const calls = {
7
+ agentTemplate: [],
8
+ credential: [],
9
+ repoMapping: [],
10
+ };
11
+ const client = {
12
+ agentTemplate: {
13
+ upsert: async (args) => {
14
+ calls.agentTemplate.push(args);
15
+ return { id: "agent-template-id", name: args.create.name };
16
+ },
17
+ },
18
+ credential: {
19
+ upsert: async (args) => {
20
+ calls.credential.push(args);
21
+ return { id: `${args.create.kind}-${args.create.name}` };
22
+ },
23
+ },
24
+ repoMapping: {
25
+ upsert: async (args) => {
26
+ calls.repoMapping.push(args);
27
+ return { id: "repo-mapping-id" };
28
+ },
29
+ },
30
+ };
31
+ return { client, calls };
32
+ }
33
+ describe("seed script", () => {
34
+ it("builds seed data that matches the validated seed shape", () => {
35
+ const seed = buildSeedData({ anthropicApiKey: "sk-ant-test" });
36
+ expect(() => SeedDataSchema.parse(seed)).not.toThrow();
37
+ expect(seed.agentTemplate).toMatchObject({
38
+ name: "default",
39
+ model: "claude-opus-4-5",
40
+ maxConcurrent: 1,
41
+ allowedTools: ["read_file", "write_file", "bash", "search"],
42
+ skills: [],
43
+ });
44
+ expect(seed.credentials.map((credential) => credential.kind)).toEqual([
45
+ "jira",
46
+ "gitlab",
47
+ "telegram",
48
+ "anthropic",
49
+ ]);
50
+ expect(seed.repoMapping).toMatchObject({
51
+ jiraProjectKey: "JAGIT",
52
+ gitlabProjectId: "your-namespace/your-repo",
53
+ defaultBaseBranch: "main",
54
+ agentTemplateName: "default",
55
+ });
56
+ });
57
+ it("rejects invalid seed data with zod", () => {
58
+ const seed = buildSeedData({ anthropicApiKey: "sk-ant-test" });
59
+ expect(() => SeedDataSchema.parse({
60
+ ...seed,
61
+ credentials: [{ ...seed.credentials[0], kind: "unknown" }],
62
+ })).toThrow();
63
+ });
64
+ it("upserts template, encrypted credentials, and repo mapping with expected unique keys", async () => {
65
+ const seed = buildSeedData({ anthropicApiKey: "sk-ant-test" });
66
+ const { client, calls } = createFakePrisma();
67
+ await seedDatabase(client, seed, KEY);
68
+ expect(calls.agentTemplate).toHaveLength(1);
69
+ expect(calls.agentTemplate[0]).toMatchObject({
70
+ where: { name: "default" },
71
+ create: {
72
+ name: "default",
73
+ model: "claude-opus-4-5",
74
+ maxConcurrent: 1,
75
+ allowedTools: ["read_file", "write_file", "bash", "search"],
76
+ skills: [],
77
+ },
78
+ update: {
79
+ model: "claude-opus-4-5",
80
+ maxConcurrent: 1,
81
+ allowedTools: ["read_file", "write_file", "bash", "search"],
82
+ skills: [],
83
+ },
84
+ });
85
+ expect(calls.credential).toHaveLength(4);
86
+ expect(calls.credential.map((call) => call.where.kind_name)).toEqual([
87
+ { kind: "jira", name: "default" },
88
+ { kind: "gitlab", name: "default" },
89
+ { kind: "telegram", name: "default" },
90
+ { kind: "anthropic", name: "default" },
91
+ ]);
92
+ const anthropicCredential = calls.credential.find((call) => call.create.kind === "anthropic");
93
+ expect(anthropicCredential.create.secrets).not.toEqual({ apiKey: "sk-ant-test" });
94
+ expect(anthropicCredential.create.secrets).toEqual(anthropicCredential.update.secrets);
95
+ expect(JSON.parse(decrypt(anthropicCredential.create.secrets.encrypted, KEY))).toEqual({
96
+ apiKey: "sk-ant-test",
97
+ });
98
+ expect(calls.repoMapping).toHaveLength(1);
99
+ expect(calls.repoMapping[0]).toMatchObject({
100
+ where: { jiraProjectKey: "JAGIT" },
101
+ create: {
102
+ jiraProjectKey: "JAGIT",
103
+ gitlabProjectId: "your-namespace/your-repo",
104
+ defaultBaseBranch: "main",
105
+ branchPrefixRules: {
106
+ Bug: "bugfix/",
107
+ Story: "feature/",
108
+ Task: "feature/",
109
+ default: "feature/",
110
+ },
111
+ agentTemplateId: "agent-template-id",
112
+ },
113
+ update: {
114
+ gitlabProjectId: "your-namespace/your-repo",
115
+ defaultBaseBranch: "main",
116
+ branchPrefixRules: {
117
+ Bug: "bugfix/",
118
+ Story: "feature/",
119
+ Task: "feature/",
120
+ default: "feature/",
121
+ },
122
+ agentTemplateId: "agent-template-id",
123
+ },
124
+ });
125
+ });
126
+ });
@@ -0,0 +1,16 @@
1
+ import { z } from "zod";
2
+ /** The BullMQ queue name — single constant used everywhere */
3
+ export declare const JOB_QUEUE = "jagit-jobs";
4
+ /** Payload pushed onto the BullMQ queue when a job is created */
5
+ export declare const JagitJobDataSchema: z.ZodObject<{
6
+ jobId: z.ZodString;
7
+ }, z.core.$strip>;
8
+ export type JagitJobData = z.infer<typeof JagitJobDataSchema>;
9
+ /** Control signals published to Redis controlChannel(jobId) */
10
+ export type ControlSignalType = "stop" | "pause" | "resume" | "approval" | "delete";
11
+ export interface ControlSignal {
12
+ type: ControlSignalType;
13
+ jobId: string;
14
+ approvalId?: string;
15
+ chosenOptionId?: string;
16
+ }
package/dist/types.js ADDED
@@ -0,0 +1,5 @@
1
+ import { z } from "zod";
2
+ /** The BullMQ queue name — single constant used everywhere */
3
+ export const JOB_QUEUE = "jagit-jobs";
4
+ /** Payload pushed onto the BullMQ queue when a job is created */
5
+ export const JagitJobDataSchema = z.object({ jobId: z.string() });
package/package.json ADDED
@@ -0,0 +1,29 @@
1
+ {
2
+ "name": "@jagit/shared",
3
+ "version": "0.0.1",
4
+ "type": "module",
5
+ "main": "dist/index.js",
6
+ "types": "dist/index.d.ts",
7
+ "files": [
8
+ "dist",
9
+ "prisma"
10
+ ],
11
+ "dependencies": {
12
+ "@prisma/adapter-pg": "^7.8.0",
13
+ "@prisma/client": "^7.8.0",
14
+ "bullmq": "^5.78.1",
15
+ "ioredis": "^5.11.1",
16
+ "pg": "^8.21.0",
17
+ "prisma": "^7.8.0",
18
+ "zod": "^4.4.3"
19
+ },
20
+ "devDependencies": {
21
+ "@types/pg": "^8.20.0"
22
+ },
23
+ "scripts": {
24
+ "postinstall": "prisma generate",
25
+ "build": "tsc -p tsconfig.json",
26
+ "typecheck": "tsc -p tsconfig.json --noEmit",
27
+ "test": "vitest run"
28
+ }
29
+ }
@@ -0,0 +1,157 @@
1
+ -- CreateSchema
2
+ CREATE SCHEMA IF NOT EXISTS "public";
3
+
4
+ -- CreateEnum
5
+ CREATE TYPE "JobStatus" AS ENUM ('queued', 'cloning', 'running', 'awaiting_approval', 'pushing', 'opening_mr', 'reporting', 'done', 'paused', 'stopped', 'failed');
6
+
7
+ -- CreateEnum
8
+ CREATE TYPE "CredentialKind" AS ENUM ('jira', 'gitlab', 'telegram', 'anthropic');
9
+
10
+ -- CreateEnum
11
+ CREATE TYPE "ApprovalStatus" AS ENUM ('pending', 'approved', 'rejected', 'expired');
12
+
13
+ -- CreateTable
14
+ CREATE TABLE "AgentTemplate" (
15
+ "id" TEXT NOT NULL,
16
+ "name" TEXT NOT NULL,
17
+ "model" TEXT NOT NULL,
18
+ "systemPrompt" TEXT NOT NULL,
19
+ "maxConcurrent" INTEGER NOT NULL DEFAULT 1,
20
+ "allowedTools" JSONB NOT NULL DEFAULT '[]',
21
+ "skills" JSONB NOT NULL DEFAULT '[]',
22
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
23
+ "updatedAt" TIMESTAMP(3) NOT NULL,
24
+
25
+ CONSTRAINT "AgentTemplate_pkey" PRIMARY KEY ("id")
26
+ );
27
+
28
+ -- CreateTable
29
+ CREATE TABLE "Credential" (
30
+ "id" TEXT NOT NULL,
31
+ "kind" "CredentialKind" NOT NULL,
32
+ "name" TEXT NOT NULL,
33
+ "secrets" JSONB NOT NULL,
34
+ "meta" JSONB NOT NULL DEFAULT '{}',
35
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
36
+ "updatedAt" TIMESTAMP(3) NOT NULL,
37
+
38
+ CONSTRAINT "Credential_pkey" PRIMARY KEY ("id")
39
+ );
40
+
41
+ -- CreateTable
42
+ CREATE TABLE "RepoMapping" (
43
+ "id" TEXT NOT NULL,
44
+ "jiraProjectKey" TEXT NOT NULL,
45
+ "gitlabProjectId" TEXT NOT NULL,
46
+ "defaultBaseBranch" TEXT NOT NULL DEFAULT 'main',
47
+ "branchPrefixRules" JSONB NOT NULL DEFAULT '{}',
48
+ "agentTemplateId" TEXT NOT NULL,
49
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
50
+ "updatedAt" TIMESTAMP(3) NOT NULL,
51
+
52
+ CONSTRAINT "RepoMapping_pkey" PRIMARY KEY ("id")
53
+ );
54
+
55
+ -- CreateTable
56
+ CREATE TABLE "Job" (
57
+ "id" TEXT NOT NULL,
58
+ "source" TEXT NOT NULL,
59
+ "jiraIssueKey" TEXT,
60
+ "gitlabProjectId" TEXT,
61
+ "branch" TEXT,
62
+ "mrUrl" TEXT,
63
+ "status" "JobStatus" NOT NULL DEFAULT 'queued',
64
+ "agentTemplateId" TEXT,
65
+ "checkpointThreadId" TEXT NOT NULL,
66
+ "dedupeKey" TEXT NOT NULL,
67
+ "tokensUsed" INTEGER NOT NULL DEFAULT 0,
68
+ "costUsd" DOUBLE PRECISION NOT NULL DEFAULT 0,
69
+ "error" TEXT,
70
+ "workdir" TEXT,
71
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
72
+ "updatedAt" TIMESTAMP(3) NOT NULL,
73
+
74
+ CONSTRAINT "Job_pkey" PRIMARY KEY ("id")
75
+ );
76
+
77
+ -- CreateTable
78
+ CREATE TABLE "JobStep" (
79
+ "id" TEXT NOT NULL,
80
+ "jobId" TEXT NOT NULL,
81
+ "name" TEXT NOT NULL,
82
+ "status" TEXT NOT NULL DEFAULT 'pending',
83
+ "detail" JSONB NOT NULL DEFAULT '{}',
84
+ "startedAt" TIMESTAMP(3),
85
+ "finishedAt" TIMESTAMP(3),
86
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
87
+
88
+ CONSTRAINT "JobStep_pkey" PRIMARY KEY ("id")
89
+ );
90
+
91
+ -- CreateTable
92
+ CREATE TABLE "JobEvent" (
93
+ "id" TEXT NOT NULL,
94
+ "jobId" TEXT NOT NULL,
95
+ "ts" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
96
+ "level" TEXT NOT NULL DEFAULT 'info',
97
+ "type" TEXT NOT NULL,
98
+ "message" TEXT NOT NULL,
99
+ "payload" JSONB NOT NULL DEFAULT '{}',
100
+
101
+ CONSTRAINT "JobEvent_pkey" PRIMARY KEY ("id")
102
+ );
103
+
104
+ -- CreateTable
105
+ CREATE TABLE "Approval" (
106
+ "id" TEXT NOT NULL,
107
+ "jobId" TEXT NOT NULL,
108
+ "stepId" TEXT,
109
+ "kind" TEXT NOT NULL,
110
+ "prompt" TEXT NOT NULL,
111
+ "options" JSONB NOT NULL DEFAULT '[]',
112
+ "status" "ApprovalStatus" NOT NULL DEFAULT 'pending',
113
+ "decidedBy" TEXT,
114
+ "decidedVia" TEXT,
115
+ "chosenOptionId" TEXT,
116
+ "telegramMessageRef" TEXT,
117
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
118
+ "decidedAt" TIMESTAMP(3),
119
+
120
+ CONSTRAINT "Approval_pkey" PRIMARY KEY ("id")
121
+ );
122
+
123
+ -- CreateIndex
124
+ CREATE UNIQUE INDEX "AgentTemplate_name_key" ON "AgentTemplate"("name");
125
+
126
+ -- CreateIndex
127
+ CREATE UNIQUE INDEX "Credential_kind_name_key" ON "Credential"("kind", "name");
128
+
129
+ -- CreateIndex
130
+ CREATE UNIQUE INDEX "RepoMapping_jiraProjectKey_key" ON "RepoMapping"("jiraProjectKey");
131
+
132
+ -- CreateIndex
133
+ CREATE UNIQUE INDEX "Job_checkpointThreadId_key" ON "Job"("checkpointThreadId");
134
+
135
+ -- CreateIndex
136
+ CREATE UNIQUE INDEX "Job_dedupeKey_key" ON "Job"("dedupeKey");
137
+
138
+ -- CreateIndex
139
+ CREATE INDEX "JobEvent_jobId_ts_idx" ON "JobEvent"("jobId", "ts");
140
+
141
+ -- CreateIndex
142
+ CREATE INDEX "Approval_jobId_status_idx" ON "Approval"("jobId", "status");
143
+
144
+ -- AddForeignKey
145
+ ALTER TABLE "RepoMapping" ADD CONSTRAINT "RepoMapping_agentTemplateId_fkey" FOREIGN KEY ("agentTemplateId") REFERENCES "AgentTemplate"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
146
+
147
+ -- AddForeignKey
148
+ ALTER TABLE "Job" ADD CONSTRAINT "Job_agentTemplateId_fkey" FOREIGN KEY ("agentTemplateId") REFERENCES "AgentTemplate"("id") ON DELETE SET NULL ON UPDATE CASCADE;
149
+
150
+ -- AddForeignKey
151
+ ALTER TABLE "JobStep" ADD CONSTRAINT "JobStep_jobId_fkey" FOREIGN KEY ("jobId") REFERENCES "Job"("id") ON DELETE CASCADE ON UPDATE CASCADE;
152
+
153
+ -- AddForeignKey
154
+ ALTER TABLE "JobEvent" ADD CONSTRAINT "JobEvent_jobId_fkey" FOREIGN KEY ("jobId") REFERENCES "Job"("id") ON DELETE CASCADE ON UPDATE CASCADE;
155
+
156
+ -- AddForeignKey
157
+ ALTER TABLE "Approval" ADD CONSTRAINT "Approval_jobId_fkey" FOREIGN KEY ("jobId") REFERENCES "Job"("id") ON DELETE CASCADE ON UPDATE CASCADE;
@@ -0,0 +1,2 @@
1
+ -- AlterTable: add missing updatedAt column to AgentTemplate
2
+ ALTER TABLE "AgentTemplate" ADD COLUMN IF NOT EXISTS "updatedAt" TIMESTAMP(3) NOT NULL DEFAULT NOW();
@@ -0,0 +1,23 @@
1
+ -- AlterTable
2
+ ALTER TABLE "AgentTemplate" ADD COLUMN "mcpServerIds" JSONB NOT NULL DEFAULT '[]';
3
+ ALTER TABLE "AgentTemplate" ADD COLUMN "requireReviewBeforeCommit" BOOLEAN NOT NULL DEFAULT true;
4
+
5
+ -- AlterTable
6
+ ALTER TABLE "Job" ADD COLUMN "reviewApprovedAt" TIMESTAMP(3);
7
+
8
+ -- CreateTable
9
+ CREATE TABLE "McpServerConfig" (
10
+ "id" TEXT NOT NULL,
11
+ "name" TEXT NOT NULL,
12
+ "command" TEXT NOT NULL,
13
+ "args" JSONB NOT NULL DEFAULT '[]',
14
+ "env" JSONB NOT NULL DEFAULT '{}',
15
+ "enabled" BOOLEAN NOT NULL DEFAULT true,
16
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
17
+ "updatedAt" TIMESTAMP(3) NOT NULL,
18
+
19
+ CONSTRAINT "McpServerConfig_pkey" PRIMARY KEY ("id")
20
+ );
21
+
22
+ -- CreateIndex
23
+ CREATE UNIQUE INDEX "McpServerConfig_name_key" ON "McpServerConfig"("name");
@@ -0,0 +1,5 @@
1
+ -- AlterTable: add HTTP transport fields to McpServerConfig
2
+ ALTER TABLE "McpServerConfig" ADD COLUMN "transport" TEXT NOT NULL DEFAULT 'stdio';
3
+ ALTER TABLE "McpServerConfig" ADD COLUMN "url" TEXT;
4
+ ALTER TABLE "McpServerConfig" ADD COLUMN "headers" JSONB NOT NULL DEFAULT '{}';
5
+ ALTER TABLE "McpServerConfig" ALTER COLUMN "command" SET DEFAULT '';
@@ -0,0 +1 @@
1
+ ALTER TABLE "AgentTemplate" ALTER COLUMN "requireReviewBeforeCommit" SET DEFAULT false;
@@ -0,0 +1,28 @@
1
+ -- CreateTable
2
+ CREATE TABLE "User" (
3
+ "id" TEXT NOT NULL,
4
+ "username" TEXT NOT NULL,
5
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
6
+
7
+ CONSTRAINT "User_pkey" PRIMARY KEY ("id")
8
+ );
9
+
10
+ -- CreateTable
11
+ CREATE TABLE "UsageUpload" (
12
+ "id" TEXT NOT NULL,
13
+ "userId" TEXT NOT NULL,
14
+ "uploadedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
15
+ "period" TEXT NOT NULL,
16
+ "data" JSONB NOT NULL,
17
+
18
+ CONSTRAINT "UsageUpload_pkey" PRIMARY KEY ("id")
19
+ );
20
+
21
+ -- CreateIndex
22
+ CREATE UNIQUE INDEX "User_username_key" ON "User"("username");
23
+
24
+ -- CreateIndex
25
+ CREATE INDEX "UsageUpload_userId_uploadedAt_idx" ON "UsageUpload"("userId", "uploadedAt");
26
+
27
+ -- AddForeignKey
28
+ ALTER TABLE "UsageUpload" ADD CONSTRAINT "UsageUpload_userId_fkey" FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
@@ -0,0 +1,35 @@
1
+ -- CreateEnum
2
+ CREATE TYPE "AgentTool" AS ENUM ('claude_code', 'codex', 'copilot');
3
+
4
+ -- CreateTable
5
+ CREATE TABLE "AgentSession" (
6
+ "id" TEXT NOT NULL,
7
+ "tool" "AgentTool" NOT NULL,
8
+ "sessionId" TEXT NOT NULL,
9
+ "userId" TEXT NOT NULL,
10
+ "model" TEXT NOT NULL,
11
+ "inputTokens" INTEGER NOT NULL DEFAULT 0,
12
+ "cachedInputTokens" INTEGER NOT NULL DEFAULT 0,
13
+ "outputTokens" INTEGER NOT NULL DEFAULT 0,
14
+ "costUsd" DOUBLE PRECISION,
15
+ "toolCallCount" INTEGER,
16
+ "startedAt" TIMESTAMP(3) NOT NULL,
17
+ "lastUpdatedAt" TIMESTAMP(3) NOT NULL,
18
+ "rawPayload" JSONB NOT NULL DEFAULT '{}',
19
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
20
+
21
+ CONSTRAINT "AgentSession_pkey" PRIMARY KEY ("id")
22
+ );
23
+
24
+ -- CreateIndex
25
+ CREATE INDEX "AgentSession_userId_lastUpdatedAt_idx" ON "AgentSession"("userId", "lastUpdatedAt");
26
+
27
+ -- CreateIndex
28
+ CREATE INDEX "AgentSession_tool_lastUpdatedAt_idx" ON "AgentSession"("tool", "lastUpdatedAt");
29
+
30
+ -- CreateIndex
31
+ CREATE UNIQUE INDEX "AgentSession_tool_sessionId_key" ON "AgentSession"("tool", "sessionId");
32
+
33
+ -- AddForeignKey
34
+ ALTER TABLE "AgentSession" ADD CONSTRAINT "AgentSession_userId_fkey" FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
35
+
@@ -0,0 +1,3 @@
1
+ # Please do not edit this file manually
2
+ # It should be added in your version-control system (i.e. Git)
3
+ provider = "postgresql"