@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.
- package/dist/ash-widget.js +2 -0
- package/dist/ash-widget.js.map +1 -0
- package/dist/embed.cjs +19 -0
- package/dist/embed.cjs.map +1 -0
- package/dist/embed.d.cts +278 -0
- package/dist/embed.d.ts +278 -0
- package/dist/embed.js +16 -0
- package/dist/embed.js.map +1 -0
- package/dist/icons.cjs +156 -0
- package/dist/icons.cjs.map +1 -0
- package/dist/icons.d.cts +1 -0
- package/dist/icons.d.ts +1 -0
- package/dist/icons.js +3 -0
- package/dist/icons.js.map +1 -0
- package/dist/index-DJwpy-R5.js +6797 -0
- package/dist/index.cjs +17502 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.cts +9590 -0
- package/dist/index.d.ts +9590 -0
- package/dist/index.js +17328 -0
- package/dist/index.js.map +1 -0
- package/dist/playground/App.d.ts +2 -0
- package/dist/playground/App.d.ts.map +1 -0
- package/dist/playground/Playground.d.ts +62 -0
- package/dist/playground/Playground.d.ts.map +1 -0
- package/dist/playground/components/ChatInput.d.ts +60 -0
- package/dist/playground/components/ChatInput.d.ts.map +1 -0
- package/dist/playground/components/MessageList.d.ts +28 -0
- package/dist/playground/components/MessageList.d.ts.map +1 -0
- package/dist/playground/components/NormalizedMessageList.d.ts +26 -0
- package/dist/playground/components/NormalizedMessageList.d.ts.map +1 -0
- package/dist/playground/components/SandboxLogsPanel.d.ts +16 -0
- package/dist/playground/components/SandboxLogsPanel.d.ts.map +1 -0
- package/dist/playground/components/Sidebar.d.ts +7 -0
- package/dist/playground/components/Sidebar.d.ts.map +1 -0
- package/dist/playground/components/ToolCallCard.d.ts +9 -0
- package/dist/playground/components/ToolCallCard.d.ts.map +1 -0
- package/dist/playground/components/icons.d.ts +9 -0
- package/dist/playground/components/icons.d.ts.map +1 -0
- package/dist/playground/contexts/ThemeContext.d.ts +9 -0
- package/dist/playground/contexts/ThemeContext.d.ts.map +1 -0
- package/dist/playground/index.d.ts +30 -0
- package/dist/playground/index.d.ts.map +1 -0
- package/dist/playground/main.d.ts +1 -0
- package/dist/playground/main.d.ts.map +1 -0
- package/dist/playground/pages/AgentsPage.d.ts +2 -0
- package/dist/playground/pages/AgentsPage.d.ts.map +1 -0
- package/dist/playground/pages/ChatPage.d.ts +2 -0
- package/dist/playground/pages/ChatPage.d.ts.map +1 -0
- package/dist/playground/pages/SessionsPage.d.ts +2 -0
- package/dist/playground/pages/SessionsPage.d.ts.map +1 -0
- package/dist/playground.css +1 -0
- package/dist/playground.d.ts +18 -0
- package/dist/playground.d.ts.map +1 -0
- package/dist/playground.js +3753 -0
- package/dist/schema-B_CVsJm5.d.cts +1585 -0
- package/dist/schema-B_CVsJm5.d.ts +1585 -0
- package/dist/schema.cjs +254 -0
- package/dist/schema.cjs.map +1 -0
- package/dist/schema.d.cts +3 -0
- package/dist/schema.d.ts +3 -0
- package/dist/schema.js +235 -0
- package/dist/schema.js.map +1 -0
- package/package.json +108 -0
package/dist/schema.cjs
ADDED
|
@@ -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';
|
package/dist/schema.d.ts
ADDED
|
@@ -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"]}
|