@ash-cloud/ash-ai 0.1.8

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 (64) hide show
  1. package/dist/ash-widget.js +2 -0
  2. package/dist/ash-widget.js.map +1 -0
  3. package/dist/embed.cjs +19 -0
  4. package/dist/embed.cjs.map +1 -0
  5. package/dist/embed.d.cts +278 -0
  6. package/dist/embed.d.ts +278 -0
  7. package/dist/embed.js +16 -0
  8. package/dist/embed.js.map +1 -0
  9. package/dist/icons.cjs +156 -0
  10. package/dist/icons.cjs.map +1 -0
  11. package/dist/icons.d.cts +1 -0
  12. package/dist/icons.d.ts +1 -0
  13. package/dist/icons.js +3 -0
  14. package/dist/icons.js.map +1 -0
  15. package/dist/index-DJwpy-R5.js +6797 -0
  16. package/dist/index.cjs +17502 -0
  17. package/dist/index.cjs.map +1 -0
  18. package/dist/index.d.cts +9590 -0
  19. package/dist/index.d.ts +9590 -0
  20. package/dist/index.js +17328 -0
  21. package/dist/index.js.map +1 -0
  22. package/dist/playground/App.d.ts +2 -0
  23. package/dist/playground/App.d.ts.map +1 -0
  24. package/dist/playground/Playground.d.ts +62 -0
  25. package/dist/playground/Playground.d.ts.map +1 -0
  26. package/dist/playground/components/ChatInput.d.ts +60 -0
  27. package/dist/playground/components/ChatInput.d.ts.map +1 -0
  28. package/dist/playground/components/MessageList.d.ts +28 -0
  29. package/dist/playground/components/MessageList.d.ts.map +1 -0
  30. package/dist/playground/components/NormalizedMessageList.d.ts +26 -0
  31. package/dist/playground/components/NormalizedMessageList.d.ts.map +1 -0
  32. package/dist/playground/components/SandboxLogsPanel.d.ts +16 -0
  33. package/dist/playground/components/SandboxLogsPanel.d.ts.map +1 -0
  34. package/dist/playground/components/Sidebar.d.ts +7 -0
  35. package/dist/playground/components/Sidebar.d.ts.map +1 -0
  36. package/dist/playground/components/ToolCallCard.d.ts +9 -0
  37. package/dist/playground/components/ToolCallCard.d.ts.map +1 -0
  38. package/dist/playground/components/icons.d.ts +9 -0
  39. package/dist/playground/components/icons.d.ts.map +1 -0
  40. package/dist/playground/contexts/ThemeContext.d.ts +9 -0
  41. package/dist/playground/contexts/ThemeContext.d.ts.map +1 -0
  42. package/dist/playground/index.d.ts +30 -0
  43. package/dist/playground/index.d.ts.map +1 -0
  44. package/dist/playground/main.d.ts +1 -0
  45. package/dist/playground/main.d.ts.map +1 -0
  46. package/dist/playground/pages/AgentsPage.d.ts +2 -0
  47. package/dist/playground/pages/AgentsPage.d.ts.map +1 -0
  48. package/dist/playground/pages/ChatPage.d.ts +2 -0
  49. package/dist/playground/pages/ChatPage.d.ts.map +1 -0
  50. package/dist/playground/pages/SessionsPage.d.ts +2 -0
  51. package/dist/playground/pages/SessionsPage.d.ts.map +1 -0
  52. package/dist/playground.css +1 -0
  53. package/dist/playground.d.ts +18 -0
  54. package/dist/playground.d.ts.map +1 -0
  55. package/dist/playground.js +3753 -0
  56. package/dist/schema-B_CVsJm5.d.cts +1585 -0
  57. package/dist/schema-B_CVsJm5.d.ts +1585 -0
  58. package/dist/schema.cjs +254 -0
  59. package/dist/schema.cjs.map +1 -0
  60. package/dist/schema.d.cts +3 -0
  61. package/dist/schema.d.ts +3 -0
  62. package/dist/schema.js +235 -0
  63. package/dist/schema.js.map +1 -0
  64. package/package.json +108 -0
@@ -0,0 +1,254 @@
1
+ 'use strict';
2
+
3
+ var pgCore = require('drizzle-orm/pg-core');
4
+ var drizzleOrm = require('drizzle-orm');
5
+
6
+ // src/storage-postgres/schema.ts
7
+ var sessionStatusEnum = pgCore.pgEnum("session_status", [
8
+ "active",
9
+ "completed",
10
+ "error",
11
+ "suspended"
12
+ ]);
13
+ var messageRoleEnum = pgCore.pgEnum("message_role", [
14
+ "user",
15
+ "assistant",
16
+ "tool",
17
+ "system"
18
+ ]);
19
+ var agentStatusEnum = pgCore.pgEnum("agent_status", [
20
+ "active",
21
+ "paused",
22
+ "deleted"
23
+ ]);
24
+ var agentBackendEnum = pgCore.pgEnum("agent_backend", [
25
+ "claude",
26
+ "gemini"
27
+ ]);
28
+ var queueItemStatusEnum = pgCore.pgEnum("queue_item_status", [
29
+ "pending",
30
+ "processing",
31
+ "completed",
32
+ "failed",
33
+ "cancelled"
34
+ ]);
35
+ var configDeploymentStatusEnum = pgCore.pgEnum("config_deployment_status", [
36
+ "pending",
37
+ "success",
38
+ "failed"
39
+ ]);
40
+ var configDeploymentTriggerEnum = pgCore.pgEnum("config_deployment_trigger", [
41
+ "webhook",
42
+ "manual",
43
+ "initial",
44
+ "rollback"
45
+ ]);
46
+ var sessions = pgCore.pgTable(
47
+ "sessions",
48
+ {
49
+ id: pgCore.uuid("id").defaultRandom().primaryKey(),
50
+ // Optional tenant isolation - null for single-tenant, set for multi-tenant
51
+ tenantId: pgCore.text("tenant_id"),
52
+ agentName: pgCore.text("agent_name").notNull(),
53
+ status: sessionStatusEnum("status").default("active").notNull(),
54
+ sdkSessionId: pgCore.text("sdk_session_id"),
55
+ // Claude Agent SDK session ID for resume
56
+ parentSessionId: pgCore.uuid("parent_session_id"),
57
+ totalTokens: pgCore.integer("total_tokens"),
58
+ totalCost: pgCore.integer("total_cost"),
59
+ // In cents
60
+ metadata: pgCore.jsonb("metadata").default({}).$type(),
61
+ createdAt: pgCore.timestamp("created_at", { withTimezone: true }).defaultNow().notNull(),
62
+ updatedAt: pgCore.timestamp("updated_at", { withTimezone: true }).defaultNow().notNull()
63
+ },
64
+ (table) => [
65
+ pgCore.index("idx_sessions_tenant").on(table.tenantId),
66
+ pgCore.index("idx_sessions_agent").on(table.agentName),
67
+ pgCore.index("idx_sessions_sdk").on(table.sdkSessionId),
68
+ pgCore.index("idx_sessions_status").on(table.status)
69
+ ]
70
+ );
71
+ var messages = pgCore.pgTable(
72
+ "messages",
73
+ {
74
+ id: pgCore.uuid("id").defaultRandom().primaryKey(),
75
+ sessionId: pgCore.uuid("session_id").notNull().references(() => sessions.id, { onDelete: "cascade" }),
76
+ role: messageRoleEnum("role").notNull(),
77
+ content: pgCore.jsonb("content").notNull().$type(),
78
+ metadata: pgCore.jsonb("metadata").default({}).$type(),
79
+ createdAt: pgCore.timestamp("created_at", { withTimezone: true }).defaultNow().notNull()
80
+ },
81
+ (table) => [
82
+ pgCore.index("idx_messages_session").on(table.sessionId, table.createdAt)
83
+ ]
84
+ );
85
+ var attachments = pgCore.pgTable(
86
+ "attachments",
87
+ {
88
+ id: pgCore.uuid("id").defaultRandom().primaryKey(),
89
+ messageId: pgCore.uuid("message_id").notNull().references(() => messages.id, { onDelete: "cascade" }),
90
+ filename: pgCore.text("filename").notNull(),
91
+ mimeType: pgCore.text("mime_type").notNull(),
92
+ size: pgCore.integer("size").notNull(),
93
+ storagePath: pgCore.text("storage_path").notNull(),
94
+ createdAt: pgCore.timestamp("created_at", { withTimezone: true }).defaultNow().notNull()
95
+ },
96
+ (table) => [pgCore.index("idx_attachments_message").on(table.messageId)]
97
+ );
98
+ var agents = pgCore.pgTable(
99
+ "agents",
100
+ {
101
+ id: pgCore.uuid("id").defaultRandom().primaryKey(),
102
+ // Optional tenant isolation - null for single-tenant, set for multi-tenant
103
+ tenantId: pgCore.text("tenant_id"),
104
+ name: pgCore.text("name").notNull(),
105
+ slug: pgCore.text("slug").notNull(),
106
+ description: pgCore.text("description"),
107
+ model: pgCore.text("model").notNull().default("claude-sonnet-4-5"),
108
+ backend: agentBackendEnum("backend").default("claude").notNull(),
109
+ systemPrompt: pgCore.text("system_prompt"),
110
+ allowedTools: pgCore.jsonb("allowed_tools").default([]).$type(),
111
+ disallowedTools: pgCore.jsonb("disallowed_tools").default([]).$type(),
112
+ maxTurns: pgCore.integer("max_turns"),
113
+ permissionMode: pgCore.text("permission_mode"),
114
+ mcpServers: pgCore.jsonb("mcp_servers").default({}).$type(),
115
+ envVars: pgCore.jsonb("env_vars").default({}).$type(),
116
+ startupScript: pgCore.text("startup_script"),
117
+ configFileUrl: pgCore.text("config_file_url"),
118
+ // .claude directory zip (skills, commands, agents, hooks, mcp)
119
+ configFileUpdatedAt: pgCore.timestamp("config_file_updated_at", { withTimezone: true }),
120
+ // When config was last updated (for cache busting)
121
+ config: pgCore.jsonb("config").default({}).$type(),
122
+ metadata: pgCore.jsonb("metadata").default({}).$type(),
123
+ status: agentStatusEnum("status").default("active").notNull(),
124
+ // Webhook configuration for execution completion notifications (outbound)
125
+ executionWebhookUrl: pgCore.text("execution_webhook_url"),
126
+ executionWebhookSecret: pgCore.text("execution_webhook_secret"),
127
+ createdAt: pgCore.timestamp("created_at", { withTimezone: true }).defaultNow().notNull(),
128
+ updatedAt: pgCore.timestamp("updated_at", { withTimezone: true }).defaultNow().notNull(),
129
+ deletedAt: pgCore.timestamp("deleted_at", { withTimezone: true })
130
+ },
131
+ (table) => [
132
+ pgCore.index("idx_agents_tenant").on(table.tenantId),
133
+ pgCore.index("idx_agents_slug").on(table.slug),
134
+ pgCore.index("idx_agents_status").on(table.status)
135
+ ]
136
+ );
137
+ var queueItems = pgCore.pgTable(
138
+ "queue_items",
139
+ {
140
+ id: pgCore.uuid("id").defaultRandom().primaryKey(),
141
+ // Optional tenant isolation - null for single-tenant, set for multi-tenant
142
+ tenantId: pgCore.text("tenant_id"),
143
+ sessionId: pgCore.uuid("session_id").references(() => sessions.id, {
144
+ onDelete: "cascade"
145
+ }),
146
+ agentName: pgCore.text("agent_name").notNull(),
147
+ prompt: pgCore.text("prompt").notNull(),
148
+ files: pgCore.jsonb("files").$type(),
149
+ sessionContext: pgCore.text("session_context"),
150
+ metadata: pgCore.jsonb("metadata").default({}).$type(),
151
+ status: queueItemStatusEnum("status").default("pending").notNull(),
152
+ priority: pgCore.integer("priority").default(0).notNull(),
153
+ retryCount: pgCore.integer("retry_count").default(0).notNull(),
154
+ maxRetries: pgCore.integer("max_retries").default(3).notNull(),
155
+ error: pgCore.text("error"),
156
+ resultMessageId: pgCore.uuid("result_message_id"),
157
+ createdAt: pgCore.timestamp("created_at", { withTimezone: true }).defaultNow().notNull(),
158
+ updatedAt: pgCore.timestamp("updated_at", { withTimezone: true }).defaultNow().notNull(),
159
+ startedAt: pgCore.timestamp("started_at", { withTimezone: true }),
160
+ completedAt: pgCore.timestamp("completed_at", { withTimezone: true })
161
+ },
162
+ (table) => [
163
+ pgCore.index("idx_queue_items_tenant").on(table.tenantId),
164
+ pgCore.index("idx_queue_items_agent").on(table.agentName),
165
+ pgCore.index("idx_queue_items_session").on(table.sessionId),
166
+ pgCore.index("idx_queue_items_status").on(table.status),
167
+ pgCore.index("idx_queue_items_pending").on(table.agentName, table.sessionId, table.status, table.priority, table.createdAt)
168
+ ]
169
+ );
170
+ var configDeployments = pgCore.pgTable(
171
+ "config_deployments",
172
+ {
173
+ id: pgCore.uuid("id").defaultRandom().primaryKey(),
174
+ agentId: pgCore.uuid("agent_id").notNull().references(() => agents.id, { onDelete: "cascade" }),
175
+ // Git commit info
176
+ commitSha: pgCore.text("commit_sha").notNull(),
177
+ commitMessage: pgCore.text("commit_message"),
178
+ commitAuthor: pgCore.text("commit_author"),
179
+ commitTimestamp: pgCore.timestamp("commit_timestamp", { withTimezone: true }),
180
+ // Deployment status
181
+ status: configDeploymentStatusEnum("status").default("pending").notNull(),
182
+ errorMessage: pgCore.text("error_message"),
183
+ // Config file info (zipped .claude directory)
184
+ configFileUrl: pgCore.text("config_file_url"),
185
+ configFileSize: pgCore.integer("config_file_size"),
186
+ // Content manifest - what's included in this deployment
187
+ manifest: pgCore.jsonb("manifest").default({}).$type(),
188
+ // Trigger info
189
+ triggeredBy: configDeploymentTriggerEnum("triggered_by").notNull(),
190
+ rollbackFromId: pgCore.uuid("rollback_from_id"),
191
+ // If this was a rollback, which deployment it came from
192
+ metadata: pgCore.jsonb("metadata").default({}).$type(),
193
+ createdAt: pgCore.timestamp("created_at", { withTimezone: true }).defaultNow().notNull()
194
+ },
195
+ (table) => [
196
+ pgCore.index("idx_config_deployments_agent").on(table.agentId),
197
+ pgCore.index("idx_config_deployments_status").on(table.status),
198
+ pgCore.index("idx_config_deployments_created").on(table.agentId, table.createdAt)
199
+ ]
200
+ );
201
+ var sessionsRelations = drizzleOrm.relations(sessions, ({ one, many }) => ({
202
+ parentSession: one(sessions, {
203
+ fields: [sessions.parentSessionId],
204
+ references: [sessions.id],
205
+ relationName: "parentSession"
206
+ }),
207
+ childSessions: many(sessions, {
208
+ relationName: "parentSession"
209
+ }),
210
+ messages: many(messages)
211
+ }));
212
+ var messagesRelations = drizzleOrm.relations(messages, ({ one, many }) => ({
213
+ session: one(sessions, {
214
+ fields: [messages.sessionId],
215
+ references: [sessions.id]
216
+ }),
217
+ attachments: many(attachments)
218
+ }));
219
+ var attachmentsRelations = drizzleOrm.relations(attachments, ({ one }) => ({
220
+ message: one(messages, {
221
+ fields: [attachments.messageId],
222
+ references: [messages.id]
223
+ })
224
+ }));
225
+ var configDeploymentsRelations = drizzleOrm.relations(configDeployments, ({ one }) => ({
226
+ agent: one(agents, {
227
+ fields: [configDeployments.agentId],
228
+ references: [agents.id]
229
+ })
230
+ }));
231
+ var agentsRelations = drizzleOrm.relations(agents, ({ many }) => ({
232
+ configDeployments: many(configDeployments)
233
+ }));
234
+
235
+ exports.agentBackendEnum = agentBackendEnum;
236
+ exports.agentStatusEnum = agentStatusEnum;
237
+ exports.agents = agents;
238
+ exports.agentsRelations = agentsRelations;
239
+ exports.attachments = attachments;
240
+ exports.attachmentsRelations = attachmentsRelations;
241
+ exports.configDeploymentStatusEnum = configDeploymentStatusEnum;
242
+ exports.configDeploymentTriggerEnum = configDeploymentTriggerEnum;
243
+ exports.configDeployments = configDeployments;
244
+ exports.configDeploymentsRelations = configDeploymentsRelations;
245
+ exports.messageRoleEnum = messageRoleEnum;
246
+ exports.messages = messages;
247
+ exports.messagesRelations = messagesRelations;
248
+ exports.queueItemStatusEnum = queueItemStatusEnum;
249
+ exports.queueItems = queueItems;
250
+ exports.sessionStatusEnum = sessionStatusEnum;
251
+ exports.sessions = sessions;
252
+ exports.sessionsRelations = sessionsRelations;
253
+ //# sourceMappingURL=schema.cjs.map
254
+ //# sourceMappingURL=schema.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/storage-postgres/schema.ts"],"names":["pgEnum","pgTable","uuid","text","integer","jsonb","timestamp","index","relations"],"mappings":"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","file":"schema.cjs","sourcesContent":["/**\n * PostgreSQL Schema for Agent SDK Harness\n *\n * This schema is designed to be:\n * 1. Compatible with @conviction-labs/harness-core types\n * 2. Tenant-agnostic - no multi-tenancy concepts in core\n * 3. Usable standalone for self-hosted deployments\n *\n * Multi-tenancy (organizations, API keys, billing) should be implemented\n * in the cloud layer by:\n * - Creating separate tables that reference these via foreign keys\n * - Or extending these tables in cloud-specific migrations\n */\n\nimport {\n pgTable,\n text,\n timestamp,\n uuid,\n jsonb,\n integer,\n index,\n pgEnum,\n} from 'drizzle-orm/pg-core';\nimport { relations } from 'drizzle-orm';\n\n// ============================================================================\n// Enums\n// ============================================================================\n\nexport const sessionStatusEnum = pgEnum('session_status', [\n 'active',\n 'completed',\n 'error',\n 'suspended',\n]);\n\nexport const messageRoleEnum = pgEnum('message_role', [\n 'user',\n 'assistant',\n 'tool',\n 'system',\n]);\n\nexport const agentStatusEnum = pgEnum('agent_status', [\n 'active',\n 'paused',\n 'deleted',\n]);\n\nexport const agentBackendEnum = pgEnum('agent_backend', [\n 'claude',\n 'gemini',\n]);\n\nexport const queueItemStatusEnum = pgEnum('queue_item_status', [\n 'pending',\n 'processing',\n 'completed',\n 'failed',\n 'cancelled',\n]);\n\n// Claude config deployment enums (for .claude directory sync from GitHub)\nexport const configDeploymentStatusEnum = pgEnum('config_deployment_status', [\n 'pending',\n 'success',\n 'failed',\n]);\n\nexport const configDeploymentTriggerEnum = pgEnum('config_deployment_trigger', [\n 'webhook',\n 'manual',\n 'initial',\n 'rollback',\n]);\n\n// ============================================================================\n// Sessions\n// ============================================================================\n\nexport const sessions = pgTable(\n 'sessions',\n {\n id: uuid('id').defaultRandom().primaryKey(),\n // Optional tenant isolation - null for single-tenant, set for multi-tenant\n tenantId: text('tenant_id'),\n agentName: text('agent_name').notNull(),\n status: sessionStatusEnum('status').default('active').notNull(),\n sdkSessionId: text('sdk_session_id'), // Claude Agent SDK session ID for resume\n parentSessionId: uuid('parent_session_id'),\n totalTokens: integer('total_tokens'),\n totalCost: integer('total_cost'), // In cents\n metadata: jsonb('metadata').default({}).$type<Record<string, unknown>>(),\n createdAt: timestamp('created_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n updatedAt: timestamp('updated_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n },\n (table) => [\n index('idx_sessions_tenant').on(table.tenantId),\n index('idx_sessions_agent').on(table.agentName),\n index('idx_sessions_sdk').on(table.sdkSessionId),\n index('idx_sessions_status').on(table.status),\n ]\n);\n\n// ============================================================================\n// Messages\n// ============================================================================\n\nexport const messages = pgTable(\n 'messages',\n {\n id: uuid('id').defaultRandom().primaryKey(),\n sessionId: uuid('session_id')\n .notNull()\n .references(() => sessions.id, { onDelete: 'cascade' }),\n role: messageRoleEnum('role').notNull(),\n content: jsonb('content').notNull().$type<unknown[]>(),\n metadata: jsonb('metadata').default({}).$type<Record<string, unknown>>(),\n createdAt: timestamp('created_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n },\n (table) => [\n index('idx_messages_session').on(table.sessionId, table.createdAt),\n ]\n);\n\n// ============================================================================\n// Attachments\n// ============================================================================\n\nexport const attachments = pgTable(\n 'attachments',\n {\n id: uuid('id').defaultRandom().primaryKey(),\n messageId: uuid('message_id')\n .notNull()\n .references(() => messages.id, { onDelete: 'cascade' }),\n filename: text('filename').notNull(),\n mimeType: text('mime_type').notNull(),\n size: integer('size').notNull(),\n storagePath: text('storage_path').notNull(),\n createdAt: timestamp('created_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n },\n (table) => [index('idx_attachments_message').on(table.messageId)]\n);\n\n// ============================================================================\n// Agents\n// ============================================================================\n\nexport const agents = pgTable(\n 'agents',\n {\n id: uuid('id').defaultRandom().primaryKey(),\n // Optional tenant isolation - null for single-tenant, set for multi-tenant\n tenantId: text('tenant_id'),\n name: text('name').notNull(),\n slug: text('slug').notNull(),\n description: text('description'),\n model: text('model').notNull().default('claude-sonnet-4-5'),\n backend: agentBackendEnum('backend').default('claude').notNull(),\n systemPrompt: text('system_prompt'),\n allowedTools: jsonb('allowed_tools').default([]).$type<string[]>(),\n disallowedTools: jsonb('disallowed_tools').default([]).$type<string[]>(),\n maxTurns: integer('max_turns'),\n permissionMode: text('permission_mode'),\n mcpServers: jsonb('mcp_servers').default({}).$type<Record<string, { command: string; args?: string[] }>>(),\n envVars: jsonb('env_vars').default({}).$type<Record<string, string>>(),\n startupScript: text('startup_script'),\n configFileUrl: text('config_file_url'), // .claude directory zip (skills, commands, agents, hooks, mcp)\n configFileUpdatedAt: timestamp('config_file_updated_at', { withTimezone: true }), // When config was last updated (for cache busting)\n config: jsonb('config').default({}).$type<Record<string, unknown>>(),\n metadata: jsonb('metadata').default({}).$type<Record<string, unknown>>(),\n status: agentStatusEnum('status').default('active').notNull(),\n // Webhook configuration for execution completion notifications (outbound)\n executionWebhookUrl: text('execution_webhook_url'),\n executionWebhookSecret: text('execution_webhook_secret'),\n createdAt: timestamp('created_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n updatedAt: timestamp('updated_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n deletedAt: timestamp('deleted_at', { withTimezone: true }),\n },\n (table) => [\n index('idx_agents_tenant').on(table.tenantId),\n index('idx_agents_slug').on(table.slug),\n index('idx_agents_status').on(table.status),\n ]\n);\n\n// ============================================================================\n// Queue Items\n// ============================================================================\n\nexport const queueItems = pgTable(\n 'queue_items',\n {\n id: uuid('id').defaultRandom().primaryKey(),\n // Optional tenant isolation - null for single-tenant, set for multi-tenant\n tenantId: text('tenant_id'),\n sessionId: uuid('session_id').references(() => sessions.id, {\n onDelete: 'cascade',\n }),\n agentName: text('agent_name').notNull(),\n prompt: text('prompt').notNull(),\n files: jsonb('files').$type<Array<{\n filename: string;\n content: string;\n mimeType: string;\n }>>(),\n sessionContext: text('session_context'),\n metadata: jsonb('metadata').default({}).$type<Record<string, unknown>>(),\n status: queueItemStatusEnum('status').default('pending').notNull(),\n priority: integer('priority').default(0).notNull(),\n retryCount: integer('retry_count').default(0).notNull(),\n maxRetries: integer('max_retries').default(3).notNull(),\n error: text('error'),\n resultMessageId: uuid('result_message_id'),\n createdAt: timestamp('created_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n updatedAt: timestamp('updated_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n startedAt: timestamp('started_at', { withTimezone: true }),\n completedAt: timestamp('completed_at', { withTimezone: true }),\n },\n (table) => [\n index('idx_queue_items_tenant').on(table.tenantId),\n index('idx_queue_items_agent').on(table.agentName),\n index('idx_queue_items_session').on(table.sessionId),\n index('idx_queue_items_status').on(table.status),\n index('idx_queue_items_pending').on(table.agentName, table.sessionId, table.status, table.priority, table.createdAt),\n ]\n);\n\n// ============================================================================\n// Config Deployments (history for GitHub-synced .claude directory)\n// ============================================================================\n\nexport const configDeployments = pgTable(\n 'config_deployments',\n {\n id: uuid('id').defaultRandom().primaryKey(),\n agentId: uuid('agent_id')\n .notNull()\n .references(() => agents.id, { onDelete: 'cascade' }),\n\n // Git commit info\n commitSha: text('commit_sha').notNull(),\n commitMessage: text('commit_message'),\n commitAuthor: text('commit_author'),\n commitTimestamp: timestamp('commit_timestamp', { withTimezone: true }),\n\n // Deployment status\n status: configDeploymentStatusEnum('status').default('pending').notNull(),\n errorMessage: text('error_message'),\n\n // Config file info (zipped .claude directory)\n configFileUrl: text('config_file_url'),\n configFileSize: integer('config_file_size'),\n\n // Content manifest - what's included in this deployment\n manifest: jsonb('manifest').default({}).$type<{\n hasSkills?: boolean;\n skillsCount?: number;\n hasCommands?: boolean;\n commandsCount?: number;\n hasAgents?: boolean;\n agentsCount?: number;\n hasHooks?: boolean;\n hasMcpConfig?: boolean;\n mcpServersCount?: number;\n }>(),\n\n // Trigger info\n triggeredBy: configDeploymentTriggerEnum('triggered_by').notNull(),\n rollbackFromId: uuid('rollback_from_id'), // If this was a rollback, which deployment it came from\n\n metadata: jsonb('metadata').default({}).$type<Record<string, unknown>>(),\n createdAt: timestamp('created_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n },\n (table) => [\n index('idx_config_deployments_agent').on(table.agentId),\n index('idx_config_deployments_status').on(table.status),\n index('idx_config_deployments_created').on(table.agentId, table.createdAt),\n ]\n);\n\n// ============================================================================\n// Relations\n// ============================================================================\n\nexport const sessionsRelations = relations(sessions, ({ one, many }) => ({\n parentSession: one(sessions, {\n fields: [sessions.parentSessionId],\n references: [sessions.id],\n relationName: 'parentSession',\n }),\n childSessions: many(sessions, {\n relationName: 'parentSession',\n }),\n messages: many(messages),\n}));\n\nexport const messagesRelations = relations(messages, ({ one, many }) => ({\n session: one(sessions, {\n fields: [messages.sessionId],\n references: [sessions.id],\n }),\n attachments: many(attachments),\n}));\n\nexport const attachmentsRelations = relations(attachments, ({ one }) => ({\n message: one(messages, {\n fields: [attachments.messageId],\n references: [messages.id],\n }),\n}));\n\nexport const configDeploymentsRelations = relations(configDeployments, ({ one }) => ({\n agent: one(agents, {\n fields: [configDeployments.agentId],\n references: [agents.id],\n }),\n}));\n\nexport const agentsRelations = relations(agents, ({ many }) => ({\n configDeployments: many(configDeployments),\n}));\n\n// ============================================================================\n// Type Exports\n// ============================================================================\n\nexport type SessionRow = typeof sessions.$inferSelect;\nexport type NewSessionRow = typeof sessions.$inferInsert;\n\nexport type MessageRow = typeof messages.$inferSelect;\nexport type NewMessageRow = typeof messages.$inferInsert;\n\nexport type AttachmentRow = typeof attachments.$inferSelect;\nexport type NewAttachmentRow = typeof attachments.$inferInsert;\n\nexport type AgentRow = typeof agents.$inferSelect;\nexport type NewAgentRow = typeof agents.$inferInsert;\n\nexport type QueueItemRow = typeof queueItems.$inferSelect;\nexport type NewQueueItemRow = typeof queueItems.$inferInsert;\n\nexport type ConfigDeploymentRow = typeof configDeployments.$inferSelect;\nexport type NewConfigDeploymentRow = typeof configDeployments.$inferInsert;\n"]}
@@ -0,0 +1,3 @@
1
+ export { w as AgentRow, A as AttachmentRow, C as ConfigDeploymentRow, M as MessageRow, x as NewAgentRow, b as NewAttachmentRow, y as NewConfigDeploymentRow, a as NewMessageRow, c as NewQueueItemRow, N as NewSessionRow, Q as QueueItemRow, S as SessionRow, f as agentBackendEnum, e as agentStatusEnum, l as agents, v as agentsRelations, k as attachments, t as attachmentsRelations, g as configDeploymentStatusEnum, h as configDeploymentTriggerEnum, o as configDeployments, u as configDeploymentsRelations, m as messageRoleEnum, j as messages, r as messagesRelations, q as queueItemStatusEnum, n as queueItems, d as sessionStatusEnum, i as sessions, p as sessionsRelations } from './schema-B_CVsJm5.cjs';
2
+ import 'drizzle-orm';
3
+ import 'drizzle-orm/pg-core';
@@ -0,0 +1,3 @@
1
+ export { w as AgentRow, A as AttachmentRow, C as ConfigDeploymentRow, M as MessageRow, x as NewAgentRow, b as NewAttachmentRow, y as NewConfigDeploymentRow, a as NewMessageRow, c as NewQueueItemRow, N as NewSessionRow, Q as QueueItemRow, S as SessionRow, f as agentBackendEnum, e as agentStatusEnum, l as agents, v as agentsRelations, k as attachments, t as attachmentsRelations, g as configDeploymentStatusEnum, h as configDeploymentTriggerEnum, o as configDeployments, u as configDeploymentsRelations, m as messageRoleEnum, j as messages, r as messagesRelations, q as queueItemStatusEnum, n as queueItems, d as sessionStatusEnum, i as sessions, p as sessionsRelations } from './schema-B_CVsJm5.js';
2
+ import 'drizzle-orm';
3
+ import 'drizzle-orm/pg-core';
package/dist/schema.js ADDED
@@ -0,0 +1,235 @@
1
+ import { pgEnum, pgTable, timestamp, jsonb, integer, uuid, text, index } from 'drizzle-orm/pg-core';
2
+ import { relations } from 'drizzle-orm';
3
+
4
+ // src/storage-postgres/schema.ts
5
+ var sessionStatusEnum = pgEnum("session_status", [
6
+ "active",
7
+ "completed",
8
+ "error",
9
+ "suspended"
10
+ ]);
11
+ var messageRoleEnum = pgEnum("message_role", [
12
+ "user",
13
+ "assistant",
14
+ "tool",
15
+ "system"
16
+ ]);
17
+ var agentStatusEnum = pgEnum("agent_status", [
18
+ "active",
19
+ "paused",
20
+ "deleted"
21
+ ]);
22
+ var agentBackendEnum = pgEnum("agent_backend", [
23
+ "claude",
24
+ "gemini"
25
+ ]);
26
+ var queueItemStatusEnum = pgEnum("queue_item_status", [
27
+ "pending",
28
+ "processing",
29
+ "completed",
30
+ "failed",
31
+ "cancelled"
32
+ ]);
33
+ var configDeploymentStatusEnum = pgEnum("config_deployment_status", [
34
+ "pending",
35
+ "success",
36
+ "failed"
37
+ ]);
38
+ var configDeploymentTriggerEnum = pgEnum("config_deployment_trigger", [
39
+ "webhook",
40
+ "manual",
41
+ "initial",
42
+ "rollback"
43
+ ]);
44
+ var sessions = pgTable(
45
+ "sessions",
46
+ {
47
+ id: uuid("id").defaultRandom().primaryKey(),
48
+ // Optional tenant isolation - null for single-tenant, set for multi-tenant
49
+ tenantId: text("tenant_id"),
50
+ agentName: text("agent_name").notNull(),
51
+ status: sessionStatusEnum("status").default("active").notNull(),
52
+ sdkSessionId: text("sdk_session_id"),
53
+ // Claude Agent SDK session ID for resume
54
+ parentSessionId: uuid("parent_session_id"),
55
+ totalTokens: integer("total_tokens"),
56
+ totalCost: integer("total_cost"),
57
+ // In cents
58
+ metadata: jsonb("metadata").default({}).$type(),
59
+ createdAt: timestamp("created_at", { withTimezone: true }).defaultNow().notNull(),
60
+ updatedAt: timestamp("updated_at", { withTimezone: true }).defaultNow().notNull()
61
+ },
62
+ (table) => [
63
+ index("idx_sessions_tenant").on(table.tenantId),
64
+ index("idx_sessions_agent").on(table.agentName),
65
+ index("idx_sessions_sdk").on(table.sdkSessionId),
66
+ index("idx_sessions_status").on(table.status)
67
+ ]
68
+ );
69
+ var messages = pgTable(
70
+ "messages",
71
+ {
72
+ id: uuid("id").defaultRandom().primaryKey(),
73
+ sessionId: uuid("session_id").notNull().references(() => sessions.id, { onDelete: "cascade" }),
74
+ role: messageRoleEnum("role").notNull(),
75
+ content: jsonb("content").notNull().$type(),
76
+ metadata: jsonb("metadata").default({}).$type(),
77
+ createdAt: timestamp("created_at", { withTimezone: true }).defaultNow().notNull()
78
+ },
79
+ (table) => [
80
+ index("idx_messages_session").on(table.sessionId, table.createdAt)
81
+ ]
82
+ );
83
+ var attachments = pgTable(
84
+ "attachments",
85
+ {
86
+ id: uuid("id").defaultRandom().primaryKey(),
87
+ messageId: uuid("message_id").notNull().references(() => messages.id, { onDelete: "cascade" }),
88
+ filename: text("filename").notNull(),
89
+ mimeType: text("mime_type").notNull(),
90
+ size: integer("size").notNull(),
91
+ storagePath: text("storage_path").notNull(),
92
+ createdAt: timestamp("created_at", { withTimezone: true }).defaultNow().notNull()
93
+ },
94
+ (table) => [index("idx_attachments_message").on(table.messageId)]
95
+ );
96
+ var agents = pgTable(
97
+ "agents",
98
+ {
99
+ id: uuid("id").defaultRandom().primaryKey(),
100
+ // Optional tenant isolation - null for single-tenant, set for multi-tenant
101
+ tenantId: text("tenant_id"),
102
+ name: text("name").notNull(),
103
+ slug: text("slug").notNull(),
104
+ description: text("description"),
105
+ model: text("model").notNull().default("claude-sonnet-4-5"),
106
+ backend: agentBackendEnum("backend").default("claude").notNull(),
107
+ systemPrompt: text("system_prompt"),
108
+ allowedTools: jsonb("allowed_tools").default([]).$type(),
109
+ disallowedTools: jsonb("disallowed_tools").default([]).$type(),
110
+ maxTurns: integer("max_turns"),
111
+ permissionMode: text("permission_mode"),
112
+ mcpServers: jsonb("mcp_servers").default({}).$type(),
113
+ envVars: jsonb("env_vars").default({}).$type(),
114
+ startupScript: text("startup_script"),
115
+ configFileUrl: text("config_file_url"),
116
+ // .claude directory zip (skills, commands, agents, hooks, mcp)
117
+ configFileUpdatedAt: timestamp("config_file_updated_at", { withTimezone: true }),
118
+ // When config was last updated (for cache busting)
119
+ config: jsonb("config").default({}).$type(),
120
+ metadata: jsonb("metadata").default({}).$type(),
121
+ status: agentStatusEnum("status").default("active").notNull(),
122
+ // Webhook configuration for execution completion notifications (outbound)
123
+ executionWebhookUrl: text("execution_webhook_url"),
124
+ executionWebhookSecret: text("execution_webhook_secret"),
125
+ createdAt: timestamp("created_at", { withTimezone: true }).defaultNow().notNull(),
126
+ updatedAt: timestamp("updated_at", { withTimezone: true }).defaultNow().notNull(),
127
+ deletedAt: timestamp("deleted_at", { withTimezone: true })
128
+ },
129
+ (table) => [
130
+ index("idx_agents_tenant").on(table.tenantId),
131
+ index("idx_agents_slug").on(table.slug),
132
+ index("idx_agents_status").on(table.status)
133
+ ]
134
+ );
135
+ var queueItems = pgTable(
136
+ "queue_items",
137
+ {
138
+ id: uuid("id").defaultRandom().primaryKey(),
139
+ // Optional tenant isolation - null for single-tenant, set for multi-tenant
140
+ tenantId: text("tenant_id"),
141
+ sessionId: uuid("session_id").references(() => sessions.id, {
142
+ onDelete: "cascade"
143
+ }),
144
+ agentName: text("agent_name").notNull(),
145
+ prompt: text("prompt").notNull(),
146
+ files: jsonb("files").$type(),
147
+ sessionContext: text("session_context"),
148
+ metadata: jsonb("metadata").default({}).$type(),
149
+ status: queueItemStatusEnum("status").default("pending").notNull(),
150
+ priority: integer("priority").default(0).notNull(),
151
+ retryCount: integer("retry_count").default(0).notNull(),
152
+ maxRetries: integer("max_retries").default(3).notNull(),
153
+ error: text("error"),
154
+ resultMessageId: uuid("result_message_id"),
155
+ createdAt: timestamp("created_at", { withTimezone: true }).defaultNow().notNull(),
156
+ updatedAt: timestamp("updated_at", { withTimezone: true }).defaultNow().notNull(),
157
+ startedAt: timestamp("started_at", { withTimezone: true }),
158
+ completedAt: timestamp("completed_at", { withTimezone: true })
159
+ },
160
+ (table) => [
161
+ index("idx_queue_items_tenant").on(table.tenantId),
162
+ index("idx_queue_items_agent").on(table.agentName),
163
+ index("idx_queue_items_session").on(table.sessionId),
164
+ index("idx_queue_items_status").on(table.status),
165
+ index("idx_queue_items_pending").on(table.agentName, table.sessionId, table.status, table.priority, table.createdAt)
166
+ ]
167
+ );
168
+ var configDeployments = pgTable(
169
+ "config_deployments",
170
+ {
171
+ id: uuid("id").defaultRandom().primaryKey(),
172
+ agentId: uuid("agent_id").notNull().references(() => agents.id, { onDelete: "cascade" }),
173
+ // Git commit info
174
+ commitSha: text("commit_sha").notNull(),
175
+ commitMessage: text("commit_message"),
176
+ commitAuthor: text("commit_author"),
177
+ commitTimestamp: timestamp("commit_timestamp", { withTimezone: true }),
178
+ // Deployment status
179
+ status: configDeploymentStatusEnum("status").default("pending").notNull(),
180
+ errorMessage: text("error_message"),
181
+ // Config file info (zipped .claude directory)
182
+ configFileUrl: text("config_file_url"),
183
+ configFileSize: integer("config_file_size"),
184
+ // Content manifest - what's included in this deployment
185
+ manifest: jsonb("manifest").default({}).$type(),
186
+ // Trigger info
187
+ triggeredBy: configDeploymentTriggerEnum("triggered_by").notNull(),
188
+ rollbackFromId: uuid("rollback_from_id"),
189
+ // If this was a rollback, which deployment it came from
190
+ metadata: jsonb("metadata").default({}).$type(),
191
+ createdAt: timestamp("created_at", { withTimezone: true }).defaultNow().notNull()
192
+ },
193
+ (table) => [
194
+ index("idx_config_deployments_agent").on(table.agentId),
195
+ index("idx_config_deployments_status").on(table.status),
196
+ index("idx_config_deployments_created").on(table.agentId, table.createdAt)
197
+ ]
198
+ );
199
+ var sessionsRelations = relations(sessions, ({ one, many }) => ({
200
+ parentSession: one(sessions, {
201
+ fields: [sessions.parentSessionId],
202
+ references: [sessions.id],
203
+ relationName: "parentSession"
204
+ }),
205
+ childSessions: many(sessions, {
206
+ relationName: "parentSession"
207
+ }),
208
+ messages: many(messages)
209
+ }));
210
+ var messagesRelations = relations(messages, ({ one, many }) => ({
211
+ session: one(sessions, {
212
+ fields: [messages.sessionId],
213
+ references: [sessions.id]
214
+ }),
215
+ attachments: many(attachments)
216
+ }));
217
+ var attachmentsRelations = relations(attachments, ({ one }) => ({
218
+ message: one(messages, {
219
+ fields: [attachments.messageId],
220
+ references: [messages.id]
221
+ })
222
+ }));
223
+ var configDeploymentsRelations = relations(configDeployments, ({ one }) => ({
224
+ agent: one(agents, {
225
+ fields: [configDeployments.agentId],
226
+ references: [agents.id]
227
+ })
228
+ }));
229
+ var agentsRelations = relations(agents, ({ many }) => ({
230
+ configDeployments: many(configDeployments)
231
+ }));
232
+
233
+ export { agentBackendEnum, agentStatusEnum, agents, agentsRelations, attachments, attachmentsRelations, configDeploymentStatusEnum, configDeploymentTriggerEnum, configDeployments, configDeploymentsRelations, messageRoleEnum, messages, messagesRelations, queueItemStatusEnum, queueItems, sessionStatusEnum, sessions, sessionsRelations };
234
+ //# sourceMappingURL=schema.js.map
235
+ //# sourceMappingURL=schema.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/storage-postgres/schema.ts"],"names":[],"mappings":"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","file":"schema.js","sourcesContent":["/**\n * PostgreSQL Schema for Agent SDK Harness\n *\n * This schema is designed to be:\n * 1. Compatible with @conviction-labs/harness-core types\n * 2. Tenant-agnostic - no multi-tenancy concepts in core\n * 3. Usable standalone for self-hosted deployments\n *\n * Multi-tenancy (organizations, API keys, billing) should be implemented\n * in the cloud layer by:\n * - Creating separate tables that reference these via foreign keys\n * - Or extending these tables in cloud-specific migrations\n */\n\nimport {\n pgTable,\n text,\n timestamp,\n uuid,\n jsonb,\n integer,\n index,\n pgEnum,\n} from 'drizzle-orm/pg-core';\nimport { relations } from 'drizzle-orm';\n\n// ============================================================================\n// Enums\n// ============================================================================\n\nexport const sessionStatusEnum = pgEnum('session_status', [\n 'active',\n 'completed',\n 'error',\n 'suspended',\n]);\n\nexport const messageRoleEnum = pgEnum('message_role', [\n 'user',\n 'assistant',\n 'tool',\n 'system',\n]);\n\nexport const agentStatusEnum = pgEnum('agent_status', [\n 'active',\n 'paused',\n 'deleted',\n]);\n\nexport const agentBackendEnum = pgEnum('agent_backend', [\n 'claude',\n 'gemini',\n]);\n\nexport const queueItemStatusEnum = pgEnum('queue_item_status', [\n 'pending',\n 'processing',\n 'completed',\n 'failed',\n 'cancelled',\n]);\n\n// Claude config deployment enums (for .claude directory sync from GitHub)\nexport const configDeploymentStatusEnum = pgEnum('config_deployment_status', [\n 'pending',\n 'success',\n 'failed',\n]);\n\nexport const configDeploymentTriggerEnum = pgEnum('config_deployment_trigger', [\n 'webhook',\n 'manual',\n 'initial',\n 'rollback',\n]);\n\n// ============================================================================\n// Sessions\n// ============================================================================\n\nexport const sessions = pgTable(\n 'sessions',\n {\n id: uuid('id').defaultRandom().primaryKey(),\n // Optional tenant isolation - null for single-tenant, set for multi-tenant\n tenantId: text('tenant_id'),\n agentName: text('agent_name').notNull(),\n status: sessionStatusEnum('status').default('active').notNull(),\n sdkSessionId: text('sdk_session_id'), // Claude Agent SDK session ID for resume\n parentSessionId: uuid('parent_session_id'),\n totalTokens: integer('total_tokens'),\n totalCost: integer('total_cost'), // In cents\n metadata: jsonb('metadata').default({}).$type<Record<string, unknown>>(),\n createdAt: timestamp('created_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n updatedAt: timestamp('updated_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n },\n (table) => [\n index('idx_sessions_tenant').on(table.tenantId),\n index('idx_sessions_agent').on(table.agentName),\n index('idx_sessions_sdk').on(table.sdkSessionId),\n index('idx_sessions_status').on(table.status),\n ]\n);\n\n// ============================================================================\n// Messages\n// ============================================================================\n\nexport const messages = pgTable(\n 'messages',\n {\n id: uuid('id').defaultRandom().primaryKey(),\n sessionId: uuid('session_id')\n .notNull()\n .references(() => sessions.id, { onDelete: 'cascade' }),\n role: messageRoleEnum('role').notNull(),\n content: jsonb('content').notNull().$type<unknown[]>(),\n metadata: jsonb('metadata').default({}).$type<Record<string, unknown>>(),\n createdAt: timestamp('created_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n },\n (table) => [\n index('idx_messages_session').on(table.sessionId, table.createdAt),\n ]\n);\n\n// ============================================================================\n// Attachments\n// ============================================================================\n\nexport const attachments = pgTable(\n 'attachments',\n {\n id: uuid('id').defaultRandom().primaryKey(),\n messageId: uuid('message_id')\n .notNull()\n .references(() => messages.id, { onDelete: 'cascade' }),\n filename: text('filename').notNull(),\n mimeType: text('mime_type').notNull(),\n size: integer('size').notNull(),\n storagePath: text('storage_path').notNull(),\n createdAt: timestamp('created_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n },\n (table) => [index('idx_attachments_message').on(table.messageId)]\n);\n\n// ============================================================================\n// Agents\n// ============================================================================\n\nexport const agents = pgTable(\n 'agents',\n {\n id: uuid('id').defaultRandom().primaryKey(),\n // Optional tenant isolation - null for single-tenant, set for multi-tenant\n tenantId: text('tenant_id'),\n name: text('name').notNull(),\n slug: text('slug').notNull(),\n description: text('description'),\n model: text('model').notNull().default('claude-sonnet-4-5'),\n backend: agentBackendEnum('backend').default('claude').notNull(),\n systemPrompt: text('system_prompt'),\n allowedTools: jsonb('allowed_tools').default([]).$type<string[]>(),\n disallowedTools: jsonb('disallowed_tools').default([]).$type<string[]>(),\n maxTurns: integer('max_turns'),\n permissionMode: text('permission_mode'),\n mcpServers: jsonb('mcp_servers').default({}).$type<Record<string, { command: string; args?: string[] }>>(),\n envVars: jsonb('env_vars').default({}).$type<Record<string, string>>(),\n startupScript: text('startup_script'),\n configFileUrl: text('config_file_url'), // .claude directory zip (skills, commands, agents, hooks, mcp)\n configFileUpdatedAt: timestamp('config_file_updated_at', { withTimezone: true }), // When config was last updated (for cache busting)\n config: jsonb('config').default({}).$type<Record<string, unknown>>(),\n metadata: jsonb('metadata').default({}).$type<Record<string, unknown>>(),\n status: agentStatusEnum('status').default('active').notNull(),\n // Webhook configuration for execution completion notifications (outbound)\n executionWebhookUrl: text('execution_webhook_url'),\n executionWebhookSecret: text('execution_webhook_secret'),\n createdAt: timestamp('created_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n updatedAt: timestamp('updated_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n deletedAt: timestamp('deleted_at', { withTimezone: true }),\n },\n (table) => [\n index('idx_agents_tenant').on(table.tenantId),\n index('idx_agents_slug').on(table.slug),\n index('idx_agents_status').on(table.status),\n ]\n);\n\n// ============================================================================\n// Queue Items\n// ============================================================================\n\nexport const queueItems = pgTable(\n 'queue_items',\n {\n id: uuid('id').defaultRandom().primaryKey(),\n // Optional tenant isolation - null for single-tenant, set for multi-tenant\n tenantId: text('tenant_id'),\n sessionId: uuid('session_id').references(() => sessions.id, {\n onDelete: 'cascade',\n }),\n agentName: text('agent_name').notNull(),\n prompt: text('prompt').notNull(),\n files: jsonb('files').$type<Array<{\n filename: string;\n content: string;\n mimeType: string;\n }>>(),\n sessionContext: text('session_context'),\n metadata: jsonb('metadata').default({}).$type<Record<string, unknown>>(),\n status: queueItemStatusEnum('status').default('pending').notNull(),\n priority: integer('priority').default(0).notNull(),\n retryCount: integer('retry_count').default(0).notNull(),\n maxRetries: integer('max_retries').default(3).notNull(),\n error: text('error'),\n resultMessageId: uuid('result_message_id'),\n createdAt: timestamp('created_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n updatedAt: timestamp('updated_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n startedAt: timestamp('started_at', { withTimezone: true }),\n completedAt: timestamp('completed_at', { withTimezone: true }),\n },\n (table) => [\n index('idx_queue_items_tenant').on(table.tenantId),\n index('idx_queue_items_agent').on(table.agentName),\n index('idx_queue_items_session').on(table.sessionId),\n index('idx_queue_items_status').on(table.status),\n index('idx_queue_items_pending').on(table.agentName, table.sessionId, table.status, table.priority, table.createdAt),\n ]\n);\n\n// ============================================================================\n// Config Deployments (history for GitHub-synced .claude directory)\n// ============================================================================\n\nexport const configDeployments = pgTable(\n 'config_deployments',\n {\n id: uuid('id').defaultRandom().primaryKey(),\n agentId: uuid('agent_id')\n .notNull()\n .references(() => agents.id, { onDelete: 'cascade' }),\n\n // Git commit info\n commitSha: text('commit_sha').notNull(),\n commitMessage: text('commit_message'),\n commitAuthor: text('commit_author'),\n commitTimestamp: timestamp('commit_timestamp', { withTimezone: true }),\n\n // Deployment status\n status: configDeploymentStatusEnum('status').default('pending').notNull(),\n errorMessage: text('error_message'),\n\n // Config file info (zipped .claude directory)\n configFileUrl: text('config_file_url'),\n configFileSize: integer('config_file_size'),\n\n // Content manifest - what's included in this deployment\n manifest: jsonb('manifest').default({}).$type<{\n hasSkills?: boolean;\n skillsCount?: number;\n hasCommands?: boolean;\n commandsCount?: number;\n hasAgents?: boolean;\n agentsCount?: number;\n hasHooks?: boolean;\n hasMcpConfig?: boolean;\n mcpServersCount?: number;\n }>(),\n\n // Trigger info\n triggeredBy: configDeploymentTriggerEnum('triggered_by').notNull(),\n rollbackFromId: uuid('rollback_from_id'), // If this was a rollback, which deployment it came from\n\n metadata: jsonb('metadata').default({}).$type<Record<string, unknown>>(),\n createdAt: timestamp('created_at', { withTimezone: true })\n .defaultNow()\n .notNull(),\n },\n (table) => [\n index('idx_config_deployments_agent').on(table.agentId),\n index('idx_config_deployments_status').on(table.status),\n index('idx_config_deployments_created').on(table.agentId, table.createdAt),\n ]\n);\n\n// ============================================================================\n// Relations\n// ============================================================================\n\nexport const sessionsRelations = relations(sessions, ({ one, many }) => ({\n parentSession: one(sessions, {\n fields: [sessions.parentSessionId],\n references: [sessions.id],\n relationName: 'parentSession',\n }),\n childSessions: many(sessions, {\n relationName: 'parentSession',\n }),\n messages: many(messages),\n}));\n\nexport const messagesRelations = relations(messages, ({ one, many }) => ({\n session: one(sessions, {\n fields: [messages.sessionId],\n references: [sessions.id],\n }),\n attachments: many(attachments),\n}));\n\nexport const attachmentsRelations = relations(attachments, ({ one }) => ({\n message: one(messages, {\n fields: [attachments.messageId],\n references: [messages.id],\n }),\n}));\n\nexport const configDeploymentsRelations = relations(configDeployments, ({ one }) => ({\n agent: one(agents, {\n fields: [configDeployments.agentId],\n references: [agents.id],\n }),\n}));\n\nexport const agentsRelations = relations(agents, ({ many }) => ({\n configDeployments: many(configDeployments),\n}));\n\n// ============================================================================\n// Type Exports\n// ============================================================================\n\nexport type SessionRow = typeof sessions.$inferSelect;\nexport type NewSessionRow = typeof sessions.$inferInsert;\n\nexport type MessageRow = typeof messages.$inferSelect;\nexport type NewMessageRow = typeof messages.$inferInsert;\n\nexport type AttachmentRow = typeof attachments.$inferSelect;\nexport type NewAttachmentRow = typeof attachments.$inferInsert;\n\nexport type AgentRow = typeof agents.$inferSelect;\nexport type NewAgentRow = typeof agents.$inferInsert;\n\nexport type QueueItemRow = typeof queueItems.$inferSelect;\nexport type NewQueueItemRow = typeof queueItems.$inferInsert;\n\nexport type ConfigDeploymentRow = typeof configDeployments.$inferSelect;\nexport type NewConfigDeploymentRow = typeof configDeployments.$inferInsert;\n"]}