@stackmemoryai/stackmemory 0.3.17 → 0.3.19
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/cli/claude-sm.js +51 -5
- package/dist/cli/claude-sm.js.map +2 -2
- package/dist/cli/codex-sm.js +52 -19
- package/dist/cli/codex-sm.js.map +2 -2
- package/dist/cli/commands/db.js +143 -0
- package/dist/cli/commands/db.js.map +7 -0
- package/dist/cli/commands/login.js +50 -0
- package/dist/cli/commands/login.js.map +7 -0
- package/dist/cli/commands/migrate.js +178 -0
- package/dist/cli/commands/migrate.js.map +7 -0
- package/dist/cli/commands/onboard.js +158 -2
- package/dist/cli/commands/onboard.js.map +2 -2
- package/dist/cli/commands/skills.js +15 -2
- package/dist/cli/commands/skills.js.map +2 -2
- package/dist/cli/index.js +118 -834
- package/dist/cli/index.js.map +3 -3
- package/dist/core/context/dual-stack-manager.js +1 -1
- package/dist/core/context/dual-stack-manager.js.map +1 -1
- package/dist/core/context/frame-database.js +1 -0
- package/dist/core/context/frame-database.js.map +2 -2
- package/dist/core/context/frame-manager.js +59 -2
- package/dist/core/context/frame-manager.js.map +2 -2
- package/dist/core/database/database-adapter.js +6 -1
- package/dist/core/database/database-adapter.js.map +2 -2
- package/dist/core/database/sqlite-adapter.js +60 -2
- package/dist/core/database/sqlite-adapter.js.map +2 -2
- package/dist/integrations/claude-code/subagent-client.js +106 -3
- package/dist/integrations/claude-code/subagent-client.js.map +2 -2
- package/dist/servers/railway/config.js +51 -0
- package/dist/servers/railway/config.js.map +7 -0
- package/dist/servers/railway/index-enhanced.js +156 -0
- package/dist/servers/railway/index-enhanced.js.map +7 -0
- package/dist/servers/railway/index.js +843 -82
- package/dist/servers/railway/index.js.map +3 -3
- package/dist/servers/railway/minimal.js +48 -3
- package/dist/servers/railway/minimal.js.map +2 -2
- package/dist/servers/railway/storage-test.js +455 -0
- package/dist/servers/railway/storage-test.js.map +7 -0
- package/dist/skills/claude-skills.js +13 -12
- package/dist/skills/claude-skills.js.map +2 -2
- package/dist/skills/recursive-agent-orchestrator.js +27 -18
- package/dist/skills/recursive-agent-orchestrator.js.map +2 -2
- package/dist/skills/unified-rlm-orchestrator.js.map +2 -2
- package/package.json +13 -21
- package/scripts/README-TESTING.md +186 -0
- package/scripts/analyze-cli-security.js +288 -0
- package/scripts/archive/add-phase-tasks-to-linear.js +163 -0
- package/scripts/archive/analyze-linear-duplicates.js +214 -0
- package/scripts/archive/analyze-remaining-duplicates.js +230 -0
- package/scripts/archive/analyze-sta-duplicates.js +292 -0
- package/scripts/archive/analyze-sta-graphql.js +399 -0
- package/scripts/archive/cancel-duplicate-tasks.ts +246 -0
- package/scripts/archive/check-all-duplicates.ts +419 -0
- package/scripts/archive/clean-duplicate-tasks.js +114 -0
- package/scripts/archive/cleanup-duplicate-tasks.ts +286 -0
- package/scripts/archive/create-phase-tasks.js +387 -0
- package/scripts/archive/delete-linear-duplicates.js +182 -0
- package/scripts/archive/delete-remaining-duplicates.js +158 -0
- package/scripts/archive/delete-sta-duplicates.js +201 -0
- package/scripts/archive/delete-sta-oauth.js +201 -0
- package/scripts/archive/export-sta-tasks.js +62 -0
- package/scripts/archive/install-auto-sync.js +266 -0
- package/scripts/archive/install-chromadb-hooks.sh +133 -0
- package/scripts/archive/install-enhanced-clear-hooks.sh +431 -0
- package/scripts/archive/install-post-task-hooks.sh +289 -0
- package/scripts/archive/install-stackmemory-hooks.sh +420 -0
- package/scripts/archive/merge-linear-duplicates-safe.ts +362 -0
- package/scripts/archive/merge-linear-duplicates.ts +180 -0
- package/scripts/archive/remove-sta-tasks.js +70 -0
- package/scripts/archive/setup-background-sync.sh +168 -0
- package/scripts/archive/setup-claude-auto-triggers.sh +181 -0
- package/scripts/archive/setup-claude-autostart.sh +305 -0
- package/scripts/archive/setup-git-hooks.sh +25 -0
- package/scripts/archive/setup-linear-oauth.sh +46 -0
- package/scripts/archive/setup-mcp.sh +113 -0
- package/scripts/archive/setup-railway-deployment.sh +81 -0
- package/scripts/auto-handoff.sh +262 -0
- package/scripts/background-sync-manager.js +416 -0
- package/scripts/benchmark-performance.ts +57 -0
- package/scripts/check-redis.ts +48 -0
- package/scripts/chromadb-auto-loader.sh +128 -0
- package/scripts/chromadb-context-loader.js +479 -0
- package/scripts/claude-chromadb-hook.js +460 -0
- package/scripts/claude-code-wrapper.sh +66 -0
- package/scripts/claude-linear-skill.js +455 -0
- package/scripts/claude-pre-commit.sh +302 -0
- package/scripts/claude-sm-autostart.js +532 -0
- package/scripts/claude-sm-setup.sh +367 -0
- package/scripts/claude-with-chromadb.sh +69 -0
- package/scripts/claude-worktree-manager.sh +323 -0
- package/scripts/claude-worktree-monitor.sh +371 -0
- package/scripts/claude-worktree-setup.sh +327 -0
- package/scripts/clean-linear-backlog.js +273 -0
- package/scripts/cleanup-old-sessions.sh +57 -0
- package/scripts/codex-wrapper.sh +88 -0
- package/scripts/create-sandbox.sh +269 -0
- package/scripts/debug-linear-update.js +174 -0
- package/scripts/delete-linear-tasks.js +167 -0
- package/scripts/deploy.sh +89 -0
- package/scripts/deployment/railway.sh +352 -0
- package/scripts/deployment/test-deployment.js +194 -0
- package/scripts/detect-and-rehydrate.js +162 -0
- package/scripts/detect-and-rehydrate.mjs +165 -0
- package/scripts/development/create-demo-tasks.js +143 -0
- package/scripts/development/debug-frame-test.js +16 -0
- package/scripts/development/demo-auto-sync.js +128 -0
- package/scripts/development/fix-all-imports.js +213 -0
- package/scripts/development/fix-imports.js +229 -0
- package/scripts/development/fix-lint-loop.cjs +103 -0
- package/scripts/development/fix-project-id.ts +161 -0
- package/scripts/development/fix-strict-mode-issues.ts +291 -0
- package/scripts/development/reorganize-structure.sh +228 -0
- package/scripts/development/test-persistence-direct.js +148 -0
- package/scripts/development/test-persistence.js +114 -0
- package/scripts/development/test-tasks.js +93 -0
- package/scripts/development/update-imports.js +212 -0
- package/scripts/fetch-linear-status.js +125 -0
- package/scripts/git-hooks/README.md +310 -0
- package/scripts/git-hooks/branch-context-manager.sh +342 -0
- package/scripts/git-hooks/post-checkout-stackmemory.sh +63 -0
- package/scripts/git-hooks/post-commit-stackmemory.sh +305 -0
- package/scripts/git-hooks/pre-commit-stackmemory.sh +275 -0
- package/scripts/hooks/cleanup-shell.sh +130 -0
- package/scripts/hooks/task-complete.sh +114 -0
- package/scripts/initialize.ts +129 -0
- package/scripts/install-claude-hooks-auto.js +104 -0
- package/scripts/install-claude-hooks.sh +133 -0
- package/scripts/install-global.sh +296 -0
- package/scripts/install.sh +235 -0
- package/scripts/linear-auto-sync.js +262 -0
- package/scripts/linear-auto-sync.sh +161 -0
- package/scripts/linear-sync-daemon.js +150 -0
- package/scripts/linear-task-review.js +237 -0
- package/scripts/list-linear-tasks.ts +178 -0
- package/scripts/mcp-proxy.js +66 -0
- package/scripts/opencode-wrapper.sh +85 -0
- package/scripts/publish-local.js +74 -0
- package/scripts/query-chromadb.ts +201 -0
- package/scripts/railway-env-setup.sh +39 -0
- package/scripts/reconcile-local-tasks.js +170 -0
- package/scripts/recreate-frames-db.js +89 -0
- package/scripts/setup/claude-integration.js +138 -0
- package/scripts/setup/configure-alias.js +125 -0
- package/scripts/setup/configure-codex-alias.js +161 -0
- package/scripts/setup/configure-opencode-alias.js +175 -0
- package/scripts/setup-claude-integration.js +204 -0
- package/scripts/setup-claude-integration.sh +183 -0
- package/scripts/setup-railway-deployment.sh +37 -0
- package/scripts/setup.sh +31 -0
- package/scripts/show-linear-summary.ts +172 -0
- package/scripts/stackmemory-auto-handoff.sh +231 -0
- package/scripts/stackmemory-daemon.sh +40 -0
- package/scripts/start-linear-sync-daemon.sh +141 -0
- package/scripts/start-temporal-paradox.sh +214 -0
- package/scripts/status.ts +159 -0
- package/scripts/sync-and-clean-tasks.js +258 -0
- package/scripts/sync-frames-from-railway.js +228 -0
- package/scripts/sync-linear-graphql.js +303 -0
- package/scripts/sync-linear-tasks.js +186 -0
- package/scripts/test-auto-triggers.sh +57 -0
- package/scripts/test-browser-mcp.js +74 -0
- package/scripts/test-chromadb-full.js +115 -0
- package/scripts/test-chromadb-hooks.sh +28 -0
- package/scripts/test-chromadb-sync.ts +245 -0
- package/scripts/test-cli-security.js +293 -0
- package/scripts/test-hooks-persistence.sh +220 -0
- package/scripts/test-installation-scenarios.sh +359 -0
- package/scripts/test-installation.sh +224 -0
- package/scripts/test-mcp.js +163 -0
- package/scripts/test-pre-publish-quick.sh +75 -0
- package/scripts/test-quality-gates.sh +263 -0
- package/scripts/test-railway-db.js +222 -0
- package/scripts/test-redis-storage.ts +490 -0
- package/scripts/test-rlm-basic.sh +122 -0
- package/scripts/test-rlm-comprehensive.sh +260 -0
- package/scripts/test-rlm-e2e.sh +268 -0
- package/scripts/test-rlm-simple.js +90 -0
- package/scripts/test-rlm.js +110 -0
- package/scripts/test-session-handoff.sh +165 -0
- package/scripts/test-shell-integration.sh +275 -0
- package/scripts/testing/ab-test-runner.ts +508 -0
- package/scripts/testing/collect-metrics.ts +457 -0
- package/scripts/testing/quick-effectiveness-demo.js +187 -0
- package/scripts/testing/real-performance-test.js +422 -0
- package/scripts/testing/run-effectiveness-tests.sh +176 -0
- package/scripts/testing/scripts/testing/ab-test-runner.js +363 -0
- package/scripts/testing/scripts/testing/collect-metrics.js +292 -0
- package/scripts/testing/simple-effectiveness-test.js +310 -0
- package/scripts/testing/src/core/context/context-bridge.js +253 -0
- package/scripts/testing/src/core/context/frame-manager.js +746 -0
- package/scripts/testing/src/core/context/shared-context-layer.js +437 -0
- package/scripts/testing/src/core/database/database-adapter.js +54 -0
- package/scripts/testing/src/core/errors/index.js +291 -0
- package/scripts/testing/src/core/errors/recovery.js +268 -0
- package/scripts/testing/src/core/monitoring/logger.js +145 -0
- package/scripts/testing/src/core/retrieval/context-retriever.js +516 -0
- package/scripts/testing/src/core/session/index.js +1 -0
- package/scripts/testing/src/core/session/session-manager.js +323 -0
- package/scripts/testing/src/core/trace/cli-trace-wrapper.js +140 -0
- package/scripts/testing/src/core/trace/db-trace-wrapper.js +251 -0
- package/scripts/testing/src/core/trace/debug-trace.js +398 -0
- package/scripts/testing/src/core/trace/index.js +120 -0
- package/scripts/testing/src/core/trace/linear-api-wrapper.js +204 -0
- package/scripts/update-linear-status.js +268 -0
- package/scripts/update-linear-tasks-fixed.js +284 -0
- package/scripts/verify-railway-schema.ts +35 -0
- package/templates/claude-hooks/hooks.json +5 -0
- package/templates/claude-hooks/on-clear.js +56 -0
- package/templates/claude-hooks/on-startup.js +56 -0
- package/templates/claude-hooks/tool-use-trace.js +67 -0
- package/dist/features/tui/components/analytics-panel.js +0 -157
- package/dist/features/tui/components/analytics-panel.js.map +0 -7
- package/dist/features/tui/components/frame-visualizer.js +0 -377
- package/dist/features/tui/components/frame-visualizer.js.map +0 -7
- package/dist/features/tui/components/pr-tracker.js +0 -135
- package/dist/features/tui/components/pr-tracker.js.map +0 -7
- package/dist/features/tui/components/session-monitor.js +0 -299
- package/dist/features/tui/components/session-monitor.js.map +0 -7
- package/dist/features/tui/components/subagent-fleet.js +0 -395
- package/dist/features/tui/components/subagent-fleet.js.map +0 -7
- package/dist/features/tui/components/task-board.js +0 -1139
- package/dist/features/tui/components/task-board.js.map +0 -7
- package/dist/features/tui/index.js +0 -408
- package/dist/features/tui/index.js.map +0 -7
- package/dist/features/tui/services/data-service.js +0 -641
- package/dist/features/tui/services/data-service.js.map +0 -7
- package/dist/features/tui/services/linear-task-reader.js +0 -102
- package/dist/features/tui/services/linear-task-reader.js.map +0 -7
- package/dist/features/tui/services/websocket-client.js +0 -162
- package/dist/features/tui/services/websocket-client.js.map +0 -7
- package/dist/features/tui/terminal-compat.js +0 -220
- package/dist/features/tui/terminal-compat.js.map +0 -7
- package/dist/features/tui/types.js +0 -1
- package/dist/features/tui/types.js.map +0 -7
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/core/database/sqlite-adapter.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * SQLite Database Adapter\n * Maintains backward compatibility with existing SQLite implementation\n */\n\nimport Database from 'better-sqlite3';\nimport {\n FeatureAwareDatabaseAdapter,\n DatabaseFeatures,\n SearchOptions,\n QueryOptions,\n AggregationOptions,\n BulkOperation,\n DatabaseStats,\n CountResult,\n VersionResult,\n FrameRow,\n EventRow,\n AnchorRow,\n} from './database-adapter.js';\nimport type { Frame, Event, Anchor } from '../context/frame-manager.js';\nimport { logger } from '../monitoring/logger.js';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\n\nexport interface SQLiteConfig {\n dbPath: string;\n walMode?: boolean;\n busyTimeout?: number;\n cacheSize?: number;\n synchronous?: 'OFF' | 'NORMAL' | 'FULL' | 'EXTRA';\n}\n\nexport class SQLiteAdapter extends FeatureAwareDatabaseAdapter {\n private db: Database.Database | null = null;\n private readonly dbPath: string;\n private inTransactionFlag = false;\n\n constructor(projectId: string, config: SQLiteConfig) {\n super(projectId, config);\n this.dbPath = config.dbPath;\n }\n\n getFeatures(): DatabaseFeatures {\n return {\n supportsFullTextSearch: false, // Could enable with FTS5\n supportsVectorSearch: false,\n supportsPartitioning: false,\n supportsAnalytics: false,\n supportsCompression: false,\n supportsMaterializedViews: false,\n supportsParallelQueries: false,\n };\n }\n\n async connect(): Promise<void> {\n if (this.db) return;\n\n const config = this.config as SQLiteConfig;\n\n // Ensure directory exists\n const dir = path.dirname(this.dbPath);\n await fs.mkdir(dir, { recursive: true });\n\n this.db = new Database(this.dbPath);\n\n // Configure SQLite for better performance\n if (config.walMode !== false) {\n this.db.pragma('journal_mode = WAL');\n }\n\n if (config.busyTimeout) {\n this.db.pragma(`busy_timeout = ${config.busyTimeout}`);\n }\n\n if (config.cacheSize) {\n this.db.pragma(`cache_size = ${config.cacheSize}`);\n }\n\n if (config.synchronous) {\n this.db.pragma(`synchronous = ${config.synchronous}`);\n }\n\n logger.info('SQLite database connected', { dbPath: this.dbPath });\n }\n\n async disconnect(): Promise<void> {\n if (!this.db) return;\n\n this.db.close();\n this.db = null;\n logger.info('SQLite database disconnected');\n }\n\n /**\n * Get raw database handle for testing purposes\n * @internal\n */\n getRawDatabase(): Database.Database | null {\n return this.db;\n }\n\n isConnected(): boolean {\n return this.db !== null && this.db.open;\n }\n\n async ping(): Promise<boolean> {\n if (!this.db) return false;\n\n try {\n this.db.prepare('SELECT 1').get();\n return true;\n } catch {\n return false;\n }\n }\n\n async initializeSchema(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS frames (\n frame_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n parent_frame_id TEXT REFERENCES frames(frame_id),\n depth INTEGER NOT NULL DEFAULT 0,\n type TEXT NOT NULL,\n name TEXT NOT NULL,\n state TEXT DEFAULT 'active',\n inputs TEXT DEFAULT '{}',\n outputs TEXT DEFAULT '{}',\n digest_text TEXT,\n digest_json TEXT DEFAULT '{}',\n created_at INTEGER DEFAULT (unixepoch()),\n closed_at INTEGER\n );\n\n CREATE TABLE IF NOT EXISTS events (\n event_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n frame_id TEXT NOT NULL,\n seq INTEGER NOT NULL,\n event_type TEXT NOT NULL,\n payload TEXT NOT NULL,\n ts INTEGER DEFAULT (unixepoch()),\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id)\n );\n\n CREATE TABLE IF NOT EXISTS anchors (\n anchor_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n type TEXT NOT NULL,\n text TEXT NOT NULL,\n priority INTEGER DEFAULT 0,\n created_at INTEGER DEFAULT (unixepoch()),\n metadata TEXT DEFAULT '{}',\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id)\n );\n\n CREATE TABLE IF NOT EXISTS schema_version (\n version INTEGER PRIMARY KEY,\n applied_at INTEGER DEFAULT (unixepoch())\n );\n\n -- Indexes for performance\n CREATE INDEX IF NOT EXISTS idx_frames_run ON frames(run_id);\n CREATE INDEX IF NOT EXISTS idx_frames_project ON frames(project_id);\n CREATE INDEX IF NOT EXISTS idx_frames_parent ON frames(parent_frame_id);\n CREATE INDEX IF NOT EXISTS idx_frames_state ON frames(state);\n CREATE INDEX IF NOT EXISTS idx_frames_created ON frames(created_at DESC);\n CREATE INDEX IF NOT EXISTS idx_events_frame ON events(frame_id);\n CREATE INDEX IF NOT EXISTS idx_events_seq ON events(frame_id, seq);\n CREATE INDEX IF NOT EXISTS idx_anchors_frame ON anchors(frame_id);\n\n -- Set initial schema version if not exists\n INSERT OR IGNORE INTO schema_version (version) VALUES (1);\n `);\n }\n\n async migrateSchema(targetVersion: number): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n const currentVersion = await this.getSchemaVersion();\n\n if (currentVersion >= targetVersion) {\n logger.info('Schema already at target version', {\n currentVersion,\n targetVersion,\n });\n return;\n }\n\n // Apply migrations sequentially\n for (let v = currentVersion + 1; v <= targetVersion; v++) {\n logger.info(`Applying migration to version ${v}`);\n // Migration logic would go here\n this.db.prepare('UPDATE schema_version SET version = ?').run(v);\n }\n }\n\n async getSchemaVersion(): Promise<number> {\n if (!this.db) throw new Error('Database not connected');\n\n try {\n const result = this.db\n .prepare('SELECT MAX(version) as version FROM schema_version')\n .get() as VersionResult;\n return result?.version || 0;\n } catch {\n return 0;\n }\n }\n\n // Frame operations\n async createFrame(frame: Partial<Frame>): Promise<string> {\n if (!this.db) throw new Error('Database not connected');\n\n const frameId = frame.frame_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO frames (\n frame_id, run_id, project_id, parent_frame_id, depth,\n type, name, state, inputs, outputs, digest_text, digest_json\n ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n frameId,\n frame.run_id,\n frame.project_id || this.projectId,\n frame.parent_frame_id || null,\n frame.depth || 0,\n frame.type,\n frame.name,\n frame.state || 'active',\n JSON.stringify(frame.inputs || {}),\n JSON.stringify(frame.outputs || {}),\n frame.digest_text || null,\n JSON.stringify(frame.digest_json || {})\n );\n\n return frameId;\n }\n\n async getFrame(frameId: string): Promise<Frame | null> {\n if (!this.db) throw new Error('Database not connected');\n\n const row = this.db\n .prepare('SELECT * FROM frames WHERE frame_id = ?')\n .get(frameId) as FrameRow | undefined;\n\n if (!row) return null;\n\n return {\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n };\n }\n\n async updateFrame(frameId: string, updates: Partial<Frame>): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n const fields = [];\n const values = [];\n\n if (updates.state !== undefined) {\n fields.push('state = ?');\n values.push(updates.state);\n }\n\n if (updates.outputs !== undefined) {\n fields.push('outputs = ?');\n values.push(JSON.stringify(updates.outputs));\n }\n\n if (updates.digest_text !== undefined) {\n fields.push('digest_text = ?');\n values.push(updates.digest_text);\n }\n\n if (updates.digest_json !== undefined) {\n fields.push('digest_json = ?');\n values.push(JSON.stringify(updates.digest_json));\n }\n\n if (updates.closed_at !== undefined) {\n fields.push('closed_at = ?');\n values.push(updates.closed_at);\n }\n\n if (fields.length === 0) return;\n\n values.push(frameId);\n\n this.db\n .prepare(\n `\n UPDATE frames SET ${fields.join(', ')} WHERE frame_id = ?\n `\n )\n .run(...values);\n }\n\n async deleteFrame(frameId: string): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n // Delete in order due to foreign keys\n await this.deleteFrameAnchors(frameId);\n await this.deleteFrameEvents(frameId);\n\n this.db.prepare('DELETE FROM frames WHERE frame_id = ?').run(frameId);\n }\n\n async getActiveFrames(runId?: string): Promise<Frame[]> {\n if (!this.db) throw new Error('Database not connected');\n\n let query = \"SELECT * FROM frames WHERE state = 'active'\";\n const params = [];\n\n if (runId) {\n query += ' AND run_id = ?';\n params.push(runId);\n }\n\n query += ' ORDER BY depth ASC, created_at ASC';\n\n const rows = this.db.prepare(query).all(...params) as any[];\n\n return rows.map((row) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n }\n\n async closeFrame(frameId: string, outputs?: any): Promise<void> {\n await this.updateFrame(frameId, {\n state: 'closed',\n outputs,\n closed_at: Date.now(),\n });\n }\n\n // Event operations\n async createEvent(event: Partial<Event>): Promise<string> {\n if (!this.db) throw new Error('Database not connected');\n\n const eventId = event.event_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO events (event_id, run_id, frame_id, seq, event_type, payload, ts)\n VALUES (?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n eventId,\n event.run_id,\n event.frame_id,\n event.seq || 0,\n event.event_type,\n JSON.stringify(event.payload || {}),\n event.ts || Date.now()\n );\n\n return eventId;\n }\n\n async getFrameEvents(\n frameId: string,\n options?: QueryOptions\n ): Promise<Event[]> {\n if (!this.db) throw new Error('Database not connected');\n\n let query = 'SELECT * FROM events WHERE frame_id = ?';\n query += this.buildOrderByClause(\n options?.orderBy || 'seq',\n options?.orderDirection\n );\n query += this.buildLimitClause(options?.limit, options?.offset);\n\n const rows = this.db.prepare(query).all(frameId) as any[];\n\n return rows.map((row) => ({\n ...row,\n payload: JSON.parse(row.payload || '{}'),\n }));\n }\n\n async deleteFrameEvents(frameId: string): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('DELETE FROM events WHERE frame_id = ?').run(frameId);\n }\n\n // Anchor operations\n async createAnchor(anchor: Partial<Anchor>): Promise<string> {\n if (!this.db) throw new Error('Database not connected');\n\n const anchorId = anchor.anchor_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO anchors (anchor_id, frame_id, project_id, type, text, priority, metadata)\n VALUES (?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n anchorId,\n anchor.frame_id,\n anchor.project_id || this.projectId,\n anchor.type,\n anchor.text,\n anchor.priority || 0,\n JSON.stringify(anchor.metadata || {})\n );\n\n return anchorId;\n }\n\n async getFrameAnchors(frameId: string): Promise<Anchor[]> {\n if (!this.db) throw new Error('Database not connected');\n\n const rows = this.db\n .prepare(\n `\n SELECT * FROM anchors WHERE frame_id = ? \n ORDER BY priority DESC, created_at ASC\n `\n )\n .all(frameId) as any[];\n\n return rows.map((row) => ({\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n }));\n }\n\n async deleteFrameAnchors(frameId: string): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('DELETE FROM anchors WHERE frame_id = ?').run(frameId);\n }\n\n // Limited search (basic LIKE queries)\n async search(\n options: SearchOptions\n ): Promise<Array<Frame & { score: number }>> {\n if (!this.db) throw new Error('Database not connected');\n\n // SQLite doesn't support HAVING on non-aggregate queries, so we filter in application\n const sql = `\n SELECT *, \n CASE \n WHEN name LIKE ? THEN 1.0\n WHEN digest_text LIKE ? THEN 0.8\n WHEN inputs LIKE ? THEN 0.6\n ELSE 0.5\n END as score\n FROM frames\n WHERE name LIKE ? OR digest_text LIKE ? OR inputs LIKE ?\n ORDER BY score DESC\n `;\n\n const params = Array(6).fill(`%${options.query}%`);\n\n let rows = this.db.prepare(sql).all(...params) as any[];\n\n // Apply score threshold in application layer\n if (options.scoreThreshold) {\n rows = rows.filter((row) => row.score >= options.scoreThreshold);\n }\n\n // Apply limit and offset in application layer if threshold is used\n if (options.limit || options.offset) {\n const start = options.offset || 0;\n const end = options.limit ? start + options.limit : rows.length;\n rows = rows.slice(start, end);\n }\n\n return rows.map((row) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n }\n\n async searchByVector(\n embedding: number[],\n options?: QueryOptions\n ): Promise<Array<Frame & { similarity: number }>> {\n // Not supported in SQLite\n logger.warn('Vector search not supported in SQLite adapter');\n return [];\n }\n\n async searchHybrid(\n textQuery: string,\n embedding: number[],\n weights?: { text: number; vector: number }\n ): Promise<Array<Frame & { score: number }>> {\n // Fall back to text search only\n return this.search({ query: textQuery, ...weights });\n }\n\n // Basic aggregation\n async aggregate(\n table: string,\n options: AggregationOptions\n ): Promise<Record<string, any>[]> {\n if (!this.db) throw new Error('Database not connected');\n\n const metrics = options.metrics\n .map(\n (m) =>\n `${m.operation}(${m.field}) AS ${m.alias || `${m.operation}_${m.field}`}`\n )\n .join(', ');\n\n let sql = `SELECT ${options.groupBy.join(', ')}, ${metrics} FROM ${table}`;\n sql += ` GROUP BY ${options.groupBy.join(', ')}`;\n\n if (options.having) {\n const havingClauses = Object.entries(options.having).map(\n ([key, value]) =>\n `${key} ${typeof value === 'object' ? value.op : '='} ?`\n );\n sql += ` HAVING ${havingClauses.join(' AND ')}`;\n }\n\n return this.db\n .prepare(sql)\n .all(...Object.values(options.having || {})) as any[];\n }\n\n // Pattern detection (basic)\n async detectPatterns(timeRange?: { start: Date; end: Date }): Promise<\n Array<{\n pattern: string;\n type: string;\n frequency: number;\n lastSeen: Date;\n }>\n > {\n if (!this.db) throw new Error('Database not connected');\n\n let sql = `\n SELECT type as pattern, type, COUNT(*) as frequency, MAX(created_at) as last_seen\n FROM frames\n `;\n\n const params = [];\n if (timeRange) {\n sql += ' WHERE created_at >= ? AND created_at <= ?';\n params.push(\n Math.floor(timeRange.start.getTime() / 1000),\n Math.floor(timeRange.end.getTime() / 1000)\n );\n }\n\n sql += ' GROUP BY type HAVING COUNT(*) > 1 ORDER BY frequency DESC';\n\n const rows = this.db.prepare(sql).all(...params) as any[];\n\n return rows.map((row) => ({\n pattern: row.pattern,\n type: row.type,\n frequency: row.frequency,\n lastSeen: new Date(row.last_seen * 1000),\n }));\n }\n\n // Bulk operations\n async executeBulk(operations: BulkOperation[]): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n await this.inTransaction(async () => {\n for (const op of operations) {\n switch (op.type) {\n case 'insert':\n // Build insert dynamically based on data\n const insertCols = Object.keys(op.data);\n const insertPlaceholders = insertCols.map(() => '?').join(',');\n this.db!.prepare(\n `INSERT INTO ${op.table} (${insertCols.join(',')}) VALUES (${insertPlaceholders})`\n ).run(...Object.values(op.data));\n break;\n\n case 'update':\n const updateSets = Object.keys(op.data)\n .map((k) => `${k} = ?`)\n .join(',');\n const whereClause = this.buildWhereClause(op.where || {});\n this.db!.prepare(\n `UPDATE ${op.table} SET ${updateSets} ${whereClause}`\n ).run(...Object.values(op.data), ...Object.values(op.where || {}));\n break;\n\n case 'delete':\n const deleteWhere = this.buildWhereClause(op.where || {});\n this.db!.prepare(`DELETE FROM ${op.table} ${deleteWhere}`).run(\n ...Object.values(op.where || {})\n );\n break;\n }\n }\n });\n }\n\n async vacuum(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.pragma('vacuum');\n logger.info('SQLite database vacuumed');\n }\n\n async analyze(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.pragma('analyze');\n logger.info('SQLite database analyzed');\n }\n\n // Statistics\n async getStats(): Promise<DatabaseStats> {\n if (!this.db) throw new Error('Database not connected');\n\n const stats = {\n totalFrames: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM frames')\n .get() as CountResult\n ).count,\n activeFrames: (\n this.db\n .prepare(\n \"SELECT COUNT(*) as count FROM frames WHERE state = 'active'\"\n )\n .get() as CountResult\n ).count,\n totalEvents: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM events')\n .get() as CountResult\n ).count,\n totalAnchors: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM anchors')\n .get() as CountResult\n ).count,\n diskUsage: 0,\n };\n\n // Get file size\n try {\n const fileStats = await fs.stat(this.dbPath);\n stats.diskUsage = fileStats.size;\n } catch {}\n\n return stats;\n }\n\n async getQueryStats(): Promise<\n Array<{\n query: string;\n calls: number;\n meanTime: number;\n totalTime: number;\n }>\n > {\n // SQLite doesn't have built-in query stats\n logger.warn('Query stats not available for SQLite');\n return [];\n }\n\n // Transaction support\n async beginTransaction(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('BEGIN').run();\n this.inTransactionFlag = true;\n }\n\n async commitTransaction(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('COMMIT').run();\n this.inTransactionFlag = false;\n }\n\n async rollbackTransaction(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('ROLLBACK').run();\n this.inTransactionFlag = false;\n }\n\n async inTransaction(\n callback: (adapter: DatabaseAdapter) => Promise<void>\n ): Promise<void> {\n await this.beginTransaction();\n\n try {\n await callback(this);\n await this.commitTransaction();\n } catch (error: unknown) {\n await this.rollbackTransaction();\n throw error;\n }\n }\n\n // Export/Import\n async exportData(\n tables: string[],\n format: 'json' | 'parquet' | 'csv'\n ): Promise<Buffer> {\n if (!this.db) throw new Error('Database not connected');\n\n if (format !== 'json') {\n throw new Error(`Format ${format} not supported for SQLite export`);\n }\n\n const data: Record<string, any[]> = {};\n\n for (const table of tables) {\n data[table] = this.db.prepare(`SELECT * FROM ${table}`).all();\n }\n\n return Buffer.from(JSON.stringify(data, null, 2));\n }\n\n async importData(\n data: Buffer,\n format: 'json' | 'parquet' | 'csv',\n options?: { truncate?: boolean; upsert?: boolean }\n ): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n if (format !== 'json') {\n throw new Error(`Format ${format} not supported for SQLite import`);\n }\n\n const parsed = JSON.parse(data.toString());\n\n await this.inTransaction(async () => {\n for (const [table, rows] of Object.entries(parsed)) {\n if (options?.truncate) {\n this.db!.prepare(`DELETE FROM ${table}`).run();\n }\n\n for (const row of rows as any[]) {\n const cols = Object.keys(row);\n const placeholders = cols.map(() => '?').join(',');\n\n if (options?.upsert) {\n const updates = cols.map((c) => `${c} = excluded.${c}`).join(',');\n this.db!.prepare(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})\n ON CONFLICT DO UPDATE SET ${updates}`\n ).run(...Object.values(row));\n } else {\n this.db!.prepare(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})`\n ).run(...Object.values(row));\n }\n }\n }\n });\n }\n}\n"],
|
|
5
|
-
"mappings": "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",
|
|
4
|
+
"sourcesContent": ["/**\n * SQLite Database Adapter\n * Maintains backward compatibility with existing SQLite implementation\n */\n\nimport Database from 'better-sqlite3';\nimport {\n FeatureAwareDatabaseAdapter,\n DatabaseFeatures,\n SearchOptions,\n QueryOptions,\n AggregationOptions,\n BulkOperation,\n DatabaseStats,\n CountResult,\n VersionResult,\n FrameRow,\n EventRow,\n AnchorRow,\n} from './database-adapter.js';\nimport type { Frame, Event, Anchor } from '../context/frame-manager.js';\nimport { logger } from '../monitoring/logger.js';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\n\nexport interface SQLiteConfig {\n dbPath: string;\n walMode?: boolean;\n busyTimeout?: number;\n cacheSize?: number;\n synchronous?: 'OFF' | 'NORMAL' | 'FULL' | 'EXTRA';\n}\n\nexport class SQLiteAdapter extends FeatureAwareDatabaseAdapter {\n private db: Database.Database | null = null;\n private readonly dbPath: string;\n private inTransactionFlag = false;\n\n constructor(projectId: string, config: SQLiteConfig) {\n super(projectId, config);\n this.dbPath = config.dbPath;\n }\n\n getFeatures(): DatabaseFeatures {\n return {\n supportsFullTextSearch: false, // Could enable with FTS5\n supportsVectorSearch: false,\n supportsPartitioning: false,\n supportsAnalytics: false,\n supportsCompression: false,\n supportsMaterializedViews: false,\n supportsParallelQueries: false,\n };\n }\n\n async connect(): Promise<void> {\n if (this.db) return;\n\n const config = this.config as SQLiteConfig;\n\n // Ensure directory exists\n const dir = path.dirname(this.dbPath);\n await fs.mkdir(dir, { recursive: true });\n\n this.db = new Database(this.dbPath);\n\n // Enforce referential integrity\n this.db.pragma('foreign_keys = ON');\n\n // Configure SQLite for better performance\n if (config.walMode !== false) {\n this.db.pragma('journal_mode = WAL');\n }\n\n if (config.busyTimeout) {\n this.db.pragma(`busy_timeout = ${config.busyTimeout}`);\n }\n\n if (config.cacheSize) {\n this.db.pragma(`cache_size = ${config.cacheSize}`);\n }\n\n if (config.synchronous) {\n this.db.pragma(`synchronous = ${config.synchronous}`);\n }\n\n logger.info('SQLite database connected', { dbPath: this.dbPath });\n }\n\n async disconnect(): Promise<void> {\n if (!this.db) return;\n\n this.db.close();\n this.db = null;\n logger.info('SQLite database disconnected');\n }\n\n /**\n * Get raw database handle for testing purposes\n * @internal\n */\n getRawDatabase(): Database.Database | null {\n return this.db;\n }\n\n isConnected(): boolean {\n return this.db !== null && this.db.open;\n }\n\n async ping(): Promise<boolean> {\n if (!this.db) return false;\n\n try {\n this.db.prepare('SELECT 1').get();\n return true;\n } catch {\n return false;\n }\n }\n\n async initializeSchema(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS frames (\n frame_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n parent_frame_id TEXT REFERENCES frames(frame_id),\n depth INTEGER NOT NULL DEFAULT 0,\n type TEXT NOT NULL,\n name TEXT NOT NULL,\n state TEXT DEFAULT 'active',\n inputs TEXT DEFAULT '{}',\n outputs TEXT DEFAULT '{}',\n digest_text TEXT,\n digest_json TEXT DEFAULT '{}',\n created_at INTEGER DEFAULT (unixepoch()),\n closed_at INTEGER\n );\n\n CREATE TABLE IF NOT EXISTS events (\n event_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n frame_id TEXT NOT NULL,\n seq INTEGER NOT NULL,\n event_type TEXT NOT NULL,\n payload TEXT NOT NULL,\n ts INTEGER DEFAULT (unixepoch()),\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );\n\n CREATE TABLE IF NOT EXISTS anchors (\n anchor_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n type TEXT NOT NULL,\n text TEXT NOT NULL,\n priority INTEGER DEFAULT 0,\n created_at INTEGER DEFAULT (unixepoch()),\n metadata TEXT DEFAULT '{}',\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );\n\n CREATE TABLE IF NOT EXISTS schema_version (\n version INTEGER PRIMARY KEY,\n applied_at INTEGER DEFAULT (unixepoch())\n );\n\n -- Indexes for performance\n CREATE INDEX IF NOT EXISTS idx_frames_run ON frames(run_id);\n CREATE INDEX IF NOT EXISTS idx_frames_project ON frames(project_id);\n CREATE INDEX IF NOT EXISTS idx_frames_parent ON frames(parent_frame_id);\n CREATE INDEX IF NOT EXISTS idx_frames_state ON frames(state);\n CREATE INDEX IF NOT EXISTS idx_frames_created ON frames(created_at DESC);\n CREATE INDEX IF NOT EXISTS idx_events_frame ON events(frame_id);\n CREATE INDEX IF NOT EXISTS idx_events_seq ON events(frame_id, seq);\n CREATE INDEX IF NOT EXISTS idx_anchors_frame ON anchors(frame_id);\n\n -- Set initial schema version if not exists\n INSERT OR IGNORE INTO schema_version (version) VALUES (1);\n `);\n\n // Ensure cascade constraints exist on dependent tables for existing DBs\n try {\n this.ensureCascadeConstraints();\n } catch (e) {\n logger.warn('Failed to ensure cascade constraints', e as Error);\n }\n }\n\n /**\n * Ensure ON DELETE CASCADE exists for events/anchors referencing frames\n * Migrates existing tables in-place if needed without data loss.\n */\n private ensureCascadeConstraints(): void {\n if (!this.db) return;\n\n const needsCascade = (table: string): boolean => {\n const rows = this.db!.prepare(`PRAGMA foreign_key_list(${table})`).all() as any[];\n // If any FK points to frames without cascade, we need migration\n return rows.some((r) => r.table === 'frames' && String(r.on_delete).toUpperCase() !== 'CASCADE');\n };\n\n const migrateTable = (table: 'events' | 'anchors') => {\n const createSql =\n table === 'events'\n ? `CREATE TABLE events_new (\n event_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n frame_id TEXT NOT NULL,\n seq INTEGER NOT NULL,\n event_type TEXT NOT NULL,\n payload TEXT NOT NULL,\n ts INTEGER DEFAULT (unixepoch()),\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );`\n : `CREATE TABLE anchors_new (\n anchor_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n type TEXT NOT NULL,\n text TEXT NOT NULL,\n priority INTEGER DEFAULT 0,\n created_at INTEGER DEFAULT (unixepoch()),\n metadata TEXT DEFAULT '{}',\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );`;\n\n const cols = table === 'events'\n ? 'event_id, run_id, frame_id, seq, event_type, payload, ts'\n : 'anchor_id, frame_id, project_id, type, text, priority, created_at, metadata';\n\n const idxSql = table === 'events'\n ? [\n 'CREATE INDEX IF NOT EXISTS idx_events_frame ON events(frame_id);',\n 'CREATE INDEX IF NOT EXISTS idx_events_seq ON events(frame_id, seq);',\n ]\n : [\n 'CREATE INDEX IF NOT EXISTS idx_anchors_frame ON anchors(frame_id);',\n ];\n\n this.db!.exec('PRAGMA foreign_keys = OFF;');\n this.db!.exec('BEGIN;');\n this.db!.exec(createSql);\n this.db!.prepare(`INSERT INTO ${table === 'events' ? 'events_new' : 'anchors_new'} (${cols}) SELECT ${cols} FROM ${table}`).run();\n this.db!.exec(`DROP TABLE ${table};`);\n this.db!.exec(`ALTER TABLE ${table}_new RENAME TO ${table};`);\n for (const stmt of idxSql) this.db!.exec(stmt);\n this.db!.exec('COMMIT;');\n this.db!.exec('PRAGMA foreign_keys = ON;');\n logger.info(`Migrated ${table} to include ON DELETE CASCADE`);\n };\n\n if (needsCascade('events')) migrateTable('events');\n if (needsCascade('anchors')) migrateTable('anchors');\n }\n\n async migrateSchema(targetVersion: number): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n const currentVersion = await this.getSchemaVersion();\n\n if (currentVersion >= targetVersion) {\n logger.info('Schema already at target version', {\n currentVersion,\n targetVersion,\n });\n return;\n }\n\n // Apply migrations sequentially\n for (let v = currentVersion + 1; v <= targetVersion; v++) {\n logger.info(`Applying migration to version ${v}`);\n // Migration logic would go here\n this.db.prepare('UPDATE schema_version SET version = ?').run(v);\n }\n }\n\n async getSchemaVersion(): Promise<number> {\n if (!this.db) throw new Error('Database not connected');\n\n try {\n const result = this.db\n .prepare('SELECT MAX(version) as version FROM schema_version')\n .get() as VersionResult;\n return result?.version || 0;\n } catch {\n return 0;\n }\n }\n\n // Frame operations\n async createFrame(frame: Partial<Frame>): Promise<string> {\n if (!this.db) throw new Error('Database not connected');\n\n const frameId = frame.frame_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO frames (\n frame_id, run_id, project_id, parent_frame_id, depth,\n type, name, state, inputs, outputs, digest_text, digest_json\n ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n frameId,\n frame.run_id,\n frame.project_id || this.projectId,\n frame.parent_frame_id || null,\n frame.depth || 0,\n frame.type,\n frame.name,\n frame.state || 'active',\n JSON.stringify(frame.inputs || {}),\n JSON.stringify(frame.outputs || {}),\n frame.digest_text || null,\n JSON.stringify(frame.digest_json || {})\n );\n\n return frameId;\n }\n\n async getFrame(frameId: string): Promise<Frame | null> {\n if (!this.db) throw new Error('Database not connected');\n\n const row = this.db\n .prepare('SELECT * FROM frames WHERE frame_id = ?')\n .get(frameId) as FrameRow | undefined;\n\n if (!row) return null;\n\n return {\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n };\n }\n\n async updateFrame(frameId: string, updates: Partial<Frame>): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n const fields = [];\n const values = [];\n\n if (updates.state !== undefined) {\n fields.push('state = ?');\n values.push(updates.state);\n }\n\n if (updates.outputs !== undefined) {\n fields.push('outputs = ?');\n values.push(JSON.stringify(updates.outputs));\n }\n\n if (updates.digest_text !== undefined) {\n fields.push('digest_text = ?');\n values.push(updates.digest_text);\n }\n\n if (updates.digest_json !== undefined) {\n fields.push('digest_json = ?');\n values.push(JSON.stringify(updates.digest_json));\n }\n\n if (updates.closed_at !== undefined) {\n fields.push('closed_at = ?');\n values.push(updates.closed_at);\n }\n\n if (fields.length === 0) return;\n\n values.push(frameId);\n\n this.db\n .prepare(\n `\n UPDATE frames SET ${fields.join(', ')} WHERE frame_id = ?\n `\n )\n .run(...values);\n }\n\n async deleteFrame(frameId: string): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n // Delete in order due to foreign keys\n await this.deleteFrameAnchors(frameId);\n await this.deleteFrameEvents(frameId);\n\n this.db.prepare('DELETE FROM frames WHERE frame_id = ?').run(frameId);\n }\n\n async getActiveFrames(runId?: string): Promise<Frame[]> {\n if (!this.db) throw new Error('Database not connected');\n\n let query = \"SELECT * FROM frames WHERE state = 'active'\";\n const params = [];\n\n if (runId) {\n query += ' AND run_id = ?';\n params.push(runId);\n }\n\n query += ' ORDER BY depth ASC, created_at ASC';\n\n const rows = this.db.prepare(query).all(...params) as any[];\n\n return rows.map((row) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n }\n\n async closeFrame(frameId: string, outputs?: any): Promise<void> {\n await this.updateFrame(frameId, {\n state: 'closed',\n outputs,\n closed_at: Date.now(),\n });\n }\n\n // Event operations\n async createEvent(event: Partial<Event>): Promise<string> {\n if (!this.db) throw new Error('Database not connected');\n\n const eventId = event.event_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO events (event_id, run_id, frame_id, seq, event_type, payload, ts)\n VALUES (?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n eventId,\n event.run_id,\n event.frame_id,\n event.seq || 0,\n event.event_type,\n JSON.stringify(event.payload || {}),\n event.ts || Date.now()\n );\n\n return eventId;\n }\n\n async getFrameEvents(\n frameId: string,\n options?: QueryOptions\n ): Promise<Event[]> {\n if (!this.db) throw new Error('Database not connected');\n\n let query = 'SELECT * FROM events WHERE frame_id = ?';\n query += this.buildOrderByClause(\n options?.orderBy || 'seq',\n options?.orderDirection\n );\n query += this.buildLimitClause(options?.limit, options?.offset);\n\n const rows = this.db.prepare(query).all(frameId) as any[];\n\n return rows.map((row) => ({\n ...row,\n payload: JSON.parse(row.payload || '{}'),\n }));\n }\n\n async deleteFrameEvents(frameId: string): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('DELETE FROM events WHERE frame_id = ?').run(frameId);\n }\n\n // Anchor operations\n async createAnchor(anchor: Partial<Anchor>): Promise<string> {\n if (!this.db) throw new Error('Database not connected');\n\n const anchorId = anchor.anchor_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO anchors (anchor_id, frame_id, project_id, type, text, priority, metadata)\n VALUES (?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n anchorId,\n anchor.frame_id,\n anchor.project_id || this.projectId,\n anchor.type,\n anchor.text,\n anchor.priority || 0,\n JSON.stringify(anchor.metadata || {})\n );\n\n return anchorId;\n }\n\n async getFrameAnchors(frameId: string): Promise<Anchor[]> {\n if (!this.db) throw new Error('Database not connected');\n\n const rows = this.db\n .prepare(\n `\n SELECT * FROM anchors WHERE frame_id = ? \n ORDER BY priority DESC, created_at ASC\n `\n )\n .all(frameId) as any[];\n\n return rows.map((row) => ({\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n }));\n }\n\n async deleteFrameAnchors(frameId: string): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('DELETE FROM anchors WHERE frame_id = ?').run(frameId);\n }\n\n // Limited search (basic LIKE queries)\n async search(\n options: SearchOptions\n ): Promise<Array<Frame & { score: number }>> {\n if (!this.db) throw new Error('Database not connected');\n\n // SQLite doesn't support HAVING on non-aggregate queries, so we filter in application\n const sql = `\n SELECT *, \n CASE \n WHEN name LIKE ? THEN 1.0\n WHEN digest_text LIKE ? THEN 0.8\n WHEN inputs LIKE ? THEN 0.6\n ELSE 0.5\n END as score\n FROM frames\n WHERE name LIKE ? OR digest_text LIKE ? OR inputs LIKE ?\n ORDER BY score DESC\n `;\n\n const params = Array(6).fill(`%${options.query}%`);\n\n let rows = this.db.prepare(sql).all(...params) as any[];\n\n // Apply score threshold in application layer\n if (options.scoreThreshold) {\n rows = rows.filter((row) => row.score >= options.scoreThreshold);\n }\n\n // Apply limit and offset in application layer if threshold is used\n if (options.limit || options.offset) {\n const start = options.offset || 0;\n const end = options.limit ? start + options.limit : rows.length;\n rows = rows.slice(start, end);\n }\n\n return rows.map((row) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n }\n\n async searchByVector(\n embedding: number[],\n options?: QueryOptions\n ): Promise<Array<Frame & { similarity: number }>> {\n // Not supported in SQLite\n logger.warn('Vector search not supported in SQLite adapter');\n return [];\n }\n\n async searchHybrid(\n textQuery: string,\n embedding: number[],\n weights?: { text: number; vector: number }\n ): Promise<Array<Frame & { score: number }>> {\n // Fall back to text search only\n return this.search({ query: textQuery, ...weights });\n }\n\n // Basic aggregation\n async aggregate(\n table: string,\n options: AggregationOptions\n ): Promise<Record<string, any>[]> {\n if (!this.db) throw new Error('Database not connected');\n\n const metrics = options.metrics\n .map(\n (m) =>\n `${m.operation}(${m.field}) AS ${m.alias || `${m.operation}_${m.field}`}`\n )\n .join(', ');\n\n let sql = `SELECT ${options.groupBy.join(', ')}, ${metrics} FROM ${table}`;\n sql += ` GROUP BY ${options.groupBy.join(', ')}`;\n\n if (options.having) {\n const havingClauses = Object.entries(options.having).map(\n ([key, value]) =>\n `${key} ${typeof value === 'object' ? value.op : '='} ?`\n );\n sql += ` HAVING ${havingClauses.join(' AND ')}`;\n }\n\n return this.db\n .prepare(sql)\n .all(...Object.values(options.having || {})) as any[];\n }\n\n // Pattern detection (basic)\n async detectPatterns(timeRange?: { start: Date; end: Date }): Promise<\n Array<{\n pattern: string;\n type: string;\n frequency: number;\n lastSeen: Date;\n }>\n > {\n if (!this.db) throw new Error('Database not connected');\n\n let sql = `\n SELECT type as pattern, type, COUNT(*) as frequency, MAX(created_at) as last_seen\n FROM frames\n `;\n\n const params = [];\n if (timeRange) {\n sql += ' WHERE created_at >= ? AND created_at <= ?';\n params.push(\n Math.floor(timeRange.start.getTime() / 1000),\n Math.floor(timeRange.end.getTime() / 1000)\n );\n }\n\n sql += ' GROUP BY type HAVING COUNT(*) > 1 ORDER BY frequency DESC';\n\n const rows = this.db.prepare(sql).all(...params) as any[];\n\n return rows.map((row) => ({\n pattern: row.pattern,\n type: row.type,\n frequency: row.frequency,\n lastSeen: new Date(row.last_seen * 1000),\n }));\n }\n\n // Bulk operations\n async executeBulk(operations: BulkOperation[]): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n await this.inTransaction(async () => {\n for (const op of operations) {\n switch (op.type) {\n case 'insert':\n // Build insert dynamically based on data\n const insertCols = Object.keys(op.data);\n const insertPlaceholders = insertCols.map(() => '?').join(',');\n this.db!.prepare(\n `INSERT INTO ${op.table} (${insertCols.join(',')}) VALUES (${insertPlaceholders})`\n ).run(...Object.values(op.data));\n break;\n\n case 'update':\n const updateSets = Object.keys(op.data)\n .map((k) => `${k} = ?`)\n .join(',');\n const whereClause = this.buildWhereClause(op.where || {});\n this.db!.prepare(\n `UPDATE ${op.table} SET ${updateSets} ${whereClause}`\n ).run(...Object.values(op.data), ...Object.values(op.where || {}));\n break;\n\n case 'delete':\n const deleteWhere = this.buildWhereClause(op.where || {});\n this.db!.prepare(`DELETE FROM ${op.table} ${deleteWhere}`).run(\n ...Object.values(op.where || {})\n );\n break;\n }\n }\n });\n }\n\n async vacuum(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.pragma('vacuum');\n logger.info('SQLite database vacuumed');\n }\n\n async analyze(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.pragma('analyze');\n logger.info('SQLite database analyzed');\n }\n\n // Statistics\n async getStats(): Promise<DatabaseStats> {\n if (!this.db) throw new Error('Database not connected');\n\n const stats = {\n totalFrames: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM frames')\n .get() as CountResult\n ).count,\n activeFrames: (\n this.db\n .prepare(\n \"SELECT COUNT(*) as count FROM frames WHERE state = 'active'\"\n )\n .get() as CountResult\n ).count,\n totalEvents: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM events')\n .get() as CountResult\n ).count,\n totalAnchors: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM anchors')\n .get() as CountResult\n ).count,\n diskUsage: 0,\n };\n\n // Get file size\n try {\n const fileStats = await fs.stat(this.dbPath);\n stats.diskUsage = fileStats.size;\n } catch {}\n\n return stats;\n }\n\n async getQueryStats(): Promise<\n Array<{\n query: string;\n calls: number;\n meanTime: number;\n totalTime: number;\n }>\n > {\n // SQLite doesn't have built-in query stats\n logger.warn('Query stats not available for SQLite');\n return [];\n }\n\n // Transaction support\n async beginTransaction(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('BEGIN').run();\n this.inTransactionFlag = true;\n }\n\n async commitTransaction(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('COMMIT').run();\n this.inTransactionFlag = false;\n }\n\n async rollbackTransaction(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('ROLLBACK').run();\n this.inTransactionFlag = false;\n }\n\n async inTransaction(\n callback: (adapter: DatabaseAdapter) => Promise<void>\n ): Promise<void> {\n await this.beginTransaction();\n\n try {\n await callback(this);\n await this.commitTransaction();\n } catch (error: unknown) {\n await this.rollbackTransaction();\n throw error;\n }\n }\n\n // Export/Import\n async exportData(\n tables: string[],\n format: 'json' | 'parquet' | 'csv'\n ): Promise<Buffer> {\n if (!this.db) throw new Error('Database not connected');\n\n if (format !== 'json') {\n throw new Error(`Format ${format} not supported for SQLite export`);\n }\n\n const data: Record<string, any[]> = {};\n\n for (const table of tables) {\n data[table] = this.db.prepare(`SELECT * FROM ${table}`).all();\n }\n\n return Buffer.from(JSON.stringify(data, null, 2));\n }\n\n async importData(\n data: Buffer,\n format: 'json' | 'parquet' | 'csv',\n options?: { truncate?: boolean; upsert?: boolean }\n ): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n if (format !== 'json') {\n throw new Error(`Format ${format} not supported for SQLite import`);\n }\n\n const parsed = JSON.parse(data.toString());\n\n await this.inTransaction(async () => {\n for (const [table, rows] of Object.entries(parsed)) {\n if (options?.truncate) {\n this.db!.prepare(`DELETE FROM ${table}`).run();\n }\n\n for (const row of rows as any[]) {\n const cols = Object.keys(row);\n const placeholders = cols.map(() => '?').join(',');\n\n if (options?.upsert) {\n const updates = cols.map((c) => `${c} = excluded.${c}`).join(',');\n this.db!.prepare(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})\n ON CONFLICT DO UPDATE SET ${updates}`\n ).run(...Object.values(row));\n } else {\n this.db!.prepare(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})`\n ).run(...Object.values(row));\n }\n }\n }\n });\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -8,13 +8,16 @@ const execAsync = promisify(exec);
|
|
|
8
8
|
class ClaudeCodeSubagentClient {
|
|
9
9
|
tempDir;
|
|
10
10
|
activeSubagents = /* @__PURE__ */ new Map();
|
|
11
|
-
|
|
11
|
+
mockMode;
|
|
12
|
+
constructor(mockMode = true) {
|
|
13
|
+
this.mockMode = mockMode;
|
|
12
14
|
this.tempDir = path.join(os.tmpdir(), "stackmemory-rlm");
|
|
13
15
|
if (!fs.existsSync(this.tempDir)) {
|
|
14
16
|
fs.mkdirSync(this.tempDir, { recursive: true });
|
|
15
17
|
}
|
|
16
18
|
logger.info("Claude Code Subagent Client initialized", {
|
|
17
|
-
tempDir: this.tempDir
|
|
19
|
+
tempDir: this.tempDir,
|
|
20
|
+
mockMode: this.mockMode
|
|
18
21
|
});
|
|
19
22
|
}
|
|
20
23
|
/**
|
|
@@ -26,8 +29,12 @@ class ClaudeCodeSubagentClient {
|
|
|
26
29
|
const subagentId = `${request.type}-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`;
|
|
27
30
|
logger.info(`Spawning ${request.type} subagent`, {
|
|
28
31
|
subagentId,
|
|
29
|
-
task: request.task.slice(0, 100)
|
|
32
|
+
task: request.task.slice(0, 100),
|
|
33
|
+
mockMode: this.mockMode
|
|
30
34
|
});
|
|
35
|
+
if (this.mockMode) {
|
|
36
|
+
return this.getMockResponse(request, startTime, subagentId);
|
|
37
|
+
}
|
|
31
38
|
try {
|
|
32
39
|
const prompt = this.buildSubagentPrompt(request);
|
|
33
40
|
const contextFile = path.join(this.tempDir, `${subagentId}-context.json`);
|
|
@@ -277,6 +284,102 @@ echo '{"status": "completed", "type": "${request.type}"}' > "${resultFile}"
|
|
|
277
284
|
throw error;
|
|
278
285
|
}
|
|
279
286
|
}
|
|
287
|
+
/**
|
|
288
|
+
* Get mock response for testing
|
|
289
|
+
*/
|
|
290
|
+
async getMockResponse(request, startTime, subagentId) {
|
|
291
|
+
await new Promise((resolve) => setTimeout(resolve, Math.random() * 20 + 10));
|
|
292
|
+
const mockResponses = {
|
|
293
|
+
planning: {
|
|
294
|
+
tasks: [
|
|
295
|
+
{ id: "task-1", name: "Analyze requirements", type: "analysis" },
|
|
296
|
+
{ id: "task-2", name: "Design solution", type: "design" },
|
|
297
|
+
{ id: "task-3", name: "Implement solution", type: "implementation" }
|
|
298
|
+
],
|
|
299
|
+
dependencies: [],
|
|
300
|
+
estimated_time: 300
|
|
301
|
+
},
|
|
302
|
+
code: {
|
|
303
|
+
implementation: `function greetUser(name: string): string {
|
|
304
|
+
if (!name || typeof name !== 'string') {
|
|
305
|
+
throw new Error('Invalid name parameter');
|
|
306
|
+
}
|
|
307
|
+
return \`Hello, \${name}!\`;
|
|
308
|
+
}`,
|
|
309
|
+
files_modified: ["src/greet.ts"],
|
|
310
|
+
lines_added: 6,
|
|
311
|
+
lines_removed: 0
|
|
312
|
+
},
|
|
313
|
+
testing: {
|
|
314
|
+
tests: [
|
|
315
|
+
{
|
|
316
|
+
name: "greetUser should return greeting",
|
|
317
|
+
code: `test('greetUser should return greeting', () => {
|
|
318
|
+
expect(greetUser('Alice')).toBe('Hello, Alice!');
|
|
319
|
+
});`,
|
|
320
|
+
type: "unit"
|
|
321
|
+
}
|
|
322
|
+
],
|
|
323
|
+
coverage: { lines: 100, branches: 100, functions: 100 }
|
|
324
|
+
},
|
|
325
|
+
linting: {
|
|
326
|
+
issues: [],
|
|
327
|
+
fixes: [],
|
|
328
|
+
passed: true
|
|
329
|
+
},
|
|
330
|
+
review: {
|
|
331
|
+
quality: 0.85,
|
|
332
|
+
issues: [
|
|
333
|
+
"Consider adding JSDoc comments",
|
|
334
|
+
"Could add more edge case tests"
|
|
335
|
+
],
|
|
336
|
+
suggestions: [
|
|
337
|
+
"Add documentation for the function",
|
|
338
|
+
"Consider adding internationalization support",
|
|
339
|
+
"Add performance tests for large inputs"
|
|
340
|
+
],
|
|
341
|
+
improvements: []
|
|
342
|
+
},
|
|
343
|
+
improve: {
|
|
344
|
+
improved_code: `/**
|
|
345
|
+
* Greets a user with their name
|
|
346
|
+
* @param name - The name of the user to greet
|
|
347
|
+
* @returns A greeting message
|
|
348
|
+
* @throws {Error} If name is invalid
|
|
349
|
+
*/
|
|
350
|
+
function greetUser(name: string): string {
|
|
351
|
+
if (!name || typeof name !== 'string') {
|
|
352
|
+
throw new Error('Invalid name parameter: name must be a non-empty string');
|
|
353
|
+
}
|
|
354
|
+
return \`Hello, \${name}!\`;
|
|
355
|
+
}`,
|
|
356
|
+
changes_made: [
|
|
357
|
+
"Added JSDoc documentation",
|
|
358
|
+
"Improved error message"
|
|
359
|
+
]
|
|
360
|
+
},
|
|
361
|
+
context: {
|
|
362
|
+
relevant_files: ["src/greet.ts", "test/greet.test.ts"],
|
|
363
|
+
patterns: ["greeting functions", "input validation"],
|
|
364
|
+
dependencies: []
|
|
365
|
+
},
|
|
366
|
+
publish: {
|
|
367
|
+
version: "1.0.0",
|
|
368
|
+
changelog: "Initial release",
|
|
369
|
+
published: false,
|
|
370
|
+
reason: "Mock mode - no actual publishing"
|
|
371
|
+
}
|
|
372
|
+
};
|
|
373
|
+
const result = mockResponses[request.type] || {};
|
|
374
|
+
return {
|
|
375
|
+
success: true,
|
|
376
|
+
result,
|
|
377
|
+
output: `Mock ${request.type} subagent completed successfully`,
|
|
378
|
+
duration: Date.now() - startTime,
|
|
379
|
+
subagentType: request.type,
|
|
380
|
+
tokens: this.estimateTokens(JSON.stringify(result))
|
|
381
|
+
};
|
|
382
|
+
}
|
|
280
383
|
/**
|
|
281
384
|
* Estimate token usage
|
|
282
385
|
*/
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/integrations/claude-code/subagent-client.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * Claude Code Subagent Client\n * \n * Uses Claude Code's Task tool to spawn subagents instead of direct API calls\n * This leverages the Claude Code Max plan for unlimited subagent execution\n */\n\nimport { logger } from '../../core/monitoring/logger.js';\nimport { exec } from 'child_process';\nimport { promisify } from 'util';\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport * as os from 'os';\n\nconst execAsync = promisify(exec);\n\nexport interface SubagentRequest {\n type: 'planning' | 'code' | 'testing' | 'linting' | 'review' | 'improve' | 'context' | 'publish';\n task: string;\n context: Record<string, any>;\n systemPrompt?: string;\n files?: string[];\n timeout?: number;\n}\n\nexport interface SubagentResponse {\n success: boolean;\n result: any;\n output?: string;\n error?: string;\n tokens?: number;\n duration: number;\n subagentType: string;\n}\n\n/**\n * Claude Code Subagent Client\n * Spawns subagents using Claude Code's Task tool\n */\nexport class ClaudeCodeSubagentClient {\n private tempDir: string;\n private activeSubagents: Map<string, AbortController> = new Map();\n \n constructor() {\n // Create temp directory for subagent communication\n this.tempDir = path.join(os.tmpdir(), 'stackmemory-rlm');\n if (!fs.existsSync(this.tempDir)) {\n fs.mkdirSync(this.tempDir, { recursive: true });\n }\n \n logger.info('Claude Code Subagent Client initialized', {\n tempDir: this.tempDir,\n });\n }\n \n /**\n * Execute a subagent task using Claude Code's Task tool\n * This will spawn a new Claude instance with specific instructions\n */\n async executeSubagent(request: SubagentRequest): Promise<SubagentResponse> {\n const startTime = Date.now();\n const subagentId = `${request.type}-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`;\n \n logger.info(`Spawning ${request.type} subagent`, {\n subagentId,\n task: request.task.slice(0, 100),\n });\n \n try {\n // Create subagent prompt based on type\n const prompt = this.buildSubagentPrompt(request);\n \n // Write context to temp file for large contexts\n const contextFile = path.join(this.tempDir, `${subagentId}-context.json`);\n await fs.promises.writeFile(\n contextFile,\n JSON.stringify(request.context, null, 2)\n );\n \n // Create result file path\n const resultFile = path.join(this.tempDir, `${subagentId}-result.json`);\n \n // Build the Task tool command\n // The Task tool will spawn a new Claude instance with the specified prompt\n const taskCommand = this.buildTaskCommand(request, prompt, contextFile, resultFile);\n \n // Execute via Claude Code's Task tool\n const result = await this.executeTaskTool(taskCommand, request.timeout);\n \n // Read result from file\n let subagentResult: any = {};\n if (fs.existsSync(resultFile)) {\n const resultContent = await fs.promises.readFile(resultFile, 'utf-8');\n try {\n subagentResult = JSON.parse(resultContent);\n } catch (e) {\n subagentResult = { rawOutput: resultContent };\n }\n }\n \n // Cleanup temp files\n this.cleanup(subagentId);\n \n return {\n success: true,\n result: subagentResult,\n output: result.stdout,\n duration: Date.now() - startTime,\n subagentType: request.type,\n tokens: this.estimateTokens(prompt + JSON.stringify(subagentResult)),\n };\n \n } catch (error: any) {\n logger.error(`Subagent execution failed: ${request.type}`, { error, subagentId });\n \n return {\n success: false,\n result: null,\n error: error.message,\n duration: Date.now() - startTime,\n subagentType: request.type,\n };\n }\n }\n \n /**\n * Execute multiple subagents in parallel\n */\n async executeParallel(requests: SubagentRequest[]): Promise<SubagentResponse[]> {\n logger.info(`Executing ${requests.length} subagents in parallel`);\n \n const promises = requests.map(request => this.executeSubagent(request));\n const results = await Promise.allSettled(promises);\n \n return results.map((result, index) => {\n if (result.status === 'fulfilled') {\n return result.value;\n } else {\n return {\n success: false,\n result: null,\n error: result.reason?.message || 'Unknown error',\n duration: 0,\n subagentType: requests[index].type,\n };\n }\n });\n }\n \n /**\n * Build subagent prompt based on type\n */\n private buildSubagentPrompt(request: SubagentRequest): string {\n const prompts: Record<string, string> = {\n planning: `You are a Planning Subagent. Your role is to decompose complex tasks into manageable subtasks.\n \n Task: ${request.task}\n \n Instructions:\n 1. Analyze the task and identify all components\n 2. Create a dependency graph of subtasks\n 3. Assign appropriate agent types to each subtask\n 4. Consider parallel execution opportunities\n 5. Include comprehensive testing at each stage\n \n Context is available in the provided file.\n \n Output a JSON structure with the task decomposition.`,\n \n code: `You are a Code Generation Subagent. Your role is to implement high-quality, production-ready code.\n \n Task: ${request.task}\n \n Instructions:\n 1. Write clean, maintainable code\n 2. Follow project conventions (check context)\n 3. Include comprehensive error handling\n 4. Add clear comments for complex logic\n 5. Ensure code is testable\n \n Context and requirements are in the provided file.\n \n Output the implementation code.`,\n \n testing: `You are a Testing Subagent specializing in comprehensive test generation.\n \n Task: ${request.task}\n \n Instructions:\n 1. Generate unit tests for all functions/methods\n 2. Create integration tests for API endpoints\n 3. Add E2E tests for critical user flows\n 4. Include edge cases and error scenarios\n 5. Ensure high code coverage (aim for 100%)\n 6. Validate that all tests pass\n \n Context and code to test are in the provided file.\n \n Output a complete test suite.`,\n \n linting: `You are a Linting Subagent ensuring code quality and standards.\n \n Task: ${request.task}\n \n Instructions:\n 1. Check for syntax errors and type issues\n 2. Verify code formatting and style\n 3. Identify security vulnerabilities\n 4. Find performance anti-patterns\n 5. Detect unused imports and dead code\n 6. Provide specific fixes for each issue\n \n Code to analyze is in the context file.\n \n Output a JSON report with issues and fixes.`,\n \n review: `You are a Code Review Subagent performing thorough multi-stage reviews.\n \n Task: ${request.task}\n \n Instructions:\n 1. Evaluate architecture and design patterns\n 2. Assess code quality and maintainability\n 3. Check performance implications\n 4. Review security considerations\n 5. Verify test coverage adequacy\n 6. Suggest specific improvements with examples\n 7. Rate quality on a 0-1 scale\n \n Code and context are in the provided file.\n \n Output a detailed review with quality score and improvements.`,\n \n improve: `You are an Improvement Subagent enhancing code based on reviews.\n \n Task: ${request.task}\n \n Instructions:\n 1. Implement all suggested improvements\n 2. Refactor for better architecture\n 3. Optimize performance bottlenecks\n 4. Enhance error handling\n 5. Improve code clarity and documentation\n 6. Add missing test cases\n 7. Ensure backward compatibility\n \n Review feedback and code are in the context file.\n \n Output the improved code.`,\n \n context: `You are a Context Retrieval Subagent finding relevant information.\n \n Task: ${request.task}\n \n Instructions:\n 1. Search project codebase for relevant code\n 2. Find similar implementations\n 3. Locate relevant documentation\n 4. Identify dependencies and patterns\n 5. Retrieve best practices\n \n Search parameters are in the context file.\n \n Output relevant context snippets.`,\n \n publish: `You are a Publishing Subagent handling releases and deployments.\n \n Task: ${request.task}\n \n Instructions:\n 1. Prepare package for publishing\n 2. Update version numbers\n 3. Generate changelog\n 4. Create GitHub release\n 5. Publish to NPM if applicable\n 6. Update documentation\n \n Release details are in the context file.\n \n Output the release plan and commands.`,\n };\n \n return request.systemPrompt || prompts[request.type] || prompts.planning;\n }\n \n /**\n * Build Task tool command\n * This creates a command that Claude Code's Task tool can execute\n */\n private buildTaskCommand(\n request: SubagentRequest,\n prompt: string,\n contextFile: string,\n resultFile: string\n ): string {\n // Create a script that the subagent will execute\n const scriptContent = `\n#!/bin/bash\n# Subagent execution script for ${request.type}\n\n# Read context\nCONTEXT=$(cat \"${contextFile}\")\n\n# Execute task based on type\ncase \"${request.type}\" in\n \"testing\")\n # For testing subagent, actually run tests\n echo \"Generating and running tests...\"\n # The subagent will generate test files and run them\n ;;\n \"linting\")\n # For linting subagent, run actual linters\n echo \"Running linters...\"\n npm run lint || true\n ;;\n \"code\")\n # For code generation, create implementation files\n echo \"Generating implementation...\"\n ;;\n *)\n # Default behavior\n echo \"Executing ${request.type} task...\"\n ;;\nesac\n\n# Write result\necho '{\"status\": \"completed\", \"type\": \"${request.type}\"}' > \"${resultFile}\"\n`;\n \n const scriptFile = path.join(this.tempDir, `${request.type}-script.sh`);\n fs.writeFileSync(scriptFile, scriptContent);\n fs.chmodSync(scriptFile, '755');\n \n // Return the command that Task tool will execute\n // In practice, this would trigger Claude Code's Task tool\n return scriptFile;\n }\n \n /**\n * Execute via Task tool (simulated for now)\n * In production, this would use Claude Code's actual Task tool API\n */\n private async executeTaskTool(\n command: string,\n timeout?: number\n ): Promise<{ stdout: string; stderr: string }> {\n try {\n // In production, this would call Claude Code's Task tool\n // For now, we simulate with a subprocess\n const result = await execAsync(command, {\n timeout: timeout || 300000, // 5 minutes default\n maxBuffer: 10 * 1024 * 1024, // 10MB buffer\n });\n \n return result;\n } catch (error: any) {\n if (error.killed || error.signal === 'SIGTERM') {\n throw new Error(`Subagent timeout after ${timeout}ms`);\n }\n throw error;\n }\n }\n \n /**\n * Estimate token usage\n */\n private estimateTokens(text: string): number {\n // Rough estimation: 1 token \u2248 4 characters\n return Math.ceil(text.length / 4);\n }\n \n /**\n * Cleanup temporary files\n */\n private cleanup(subagentId: string): void {\n const patterns = [\n `${subagentId}-context.json`,\n `${subagentId}-result.json`,\n `${subagentId}-script.sh`,\n ];\n \n for (const pattern of patterns) {\n const filePath = path.join(this.tempDir, pattern);\n if (fs.existsSync(filePath)) {\n try {\n fs.unlinkSync(filePath);\n } catch (e) {\n // Ignore cleanup errors\n }\n }\n }\n }\n \n /**\n * Create a mock Task tool response for development\n * This simulates what Claude Code's Task tool would return\n */\n async mockTaskToolExecution(request: SubagentRequest): Promise<SubagentResponse> {\n const startTime = Date.now();\n \n // Simulate processing delay\n await new Promise(resolve => setTimeout(resolve, 1000 + Math.random() * 2000));\n \n // Generate mock responses based on subagent type\n const mockResponses: Record<string, any> = {\n planning: {\n tasks: [\n { id: '1', type: 'analyze', description: 'Analyze requirements' },\n { id: '2', type: 'implement', description: 'Implement solution' },\n { id: '3', type: 'test', description: 'Test implementation' },\n { id: '4', type: 'review', description: 'Review and improve' },\n ],\n dependencies: { '2': ['1'], '3': ['2'], '4': ['3'] },\n },\n code: {\n implementation: `\nexport class Solution {\n constructor(private config: any) {}\n \n async execute(input: string): Promise<string> {\n // Implementation generated by Code subagent\n return this.process(input);\n }\n \n private process(input: string): string {\n return \\`Processed: \\${input}\\`;\n }\n}`,\n files: ['src/solution.ts'],\n },\n testing: {\n tests: `\ndescribe('Solution', () => {\n it('should process input correctly', () => {\n const solution = new Solution({});\n const result = solution.execute('test');\n expect(result).toBe('Processed: test');\n });\n \n it('should handle edge cases', () => {\n // Edge case tests\n });\n});`,\n coverage: { lines: 95, branches: 88, functions: 100 },\n },\n review: {\n quality: 0.82,\n issues: [\n { severity: 'high', message: 'Missing error handling' },\n { severity: 'medium', message: 'Could improve type safety' },\n ],\n suggestions: [\n 'Add try-catch blocks',\n 'Use stricter TypeScript types',\n 'Add input validation',\n ],\n },\n };\n \n return {\n success: true,\n result: mockResponses[request.type] || { status: 'completed' },\n output: `Mock ${request.type} subagent completed successfully`,\n duration: Date.now() - startTime,\n subagentType: request.type,\n tokens: Math.floor(Math.random() * 5000) + 1000,\n };\n }\n \n /**\n * Get active subagent statistics\n */\n getStats() {\n return {\n activeSubagents: this.activeSubagents.size,\n tempDir: this.tempDir,\n };\n }\n \n /**\n * Cleanup all resources\n */\n async cleanupAll(): Promise<void> {\n // Abort all active subagents\n for (const [id, controller] of this.activeSubagents) {\n controller.abort();\n }\n this.activeSubagents.clear();\n \n // Clean temp directory\n if (fs.existsSync(this.tempDir)) {\n const files = await fs.promises.readdir(this.tempDir);\n for (const file of files) {\n await fs.promises.unlink(path.join(this.tempDir, file));\n }\n }\n \n logger.info('Claude Code Subagent Client cleaned up');\n }\n}"],
|
|
5
|
-
"mappings": "AAOA,SAAS,cAAc;AACvB,SAAS,YAAY;AACrB,SAAS,iBAAiB;AAC1B,YAAY,QAAQ;AACpB,YAAY,UAAU;AACtB,YAAY,QAAQ;AAEpB,MAAM,YAAY,UAAU,IAAI;AAyBzB,MAAM,yBAAyB;AAAA,EAC5B;AAAA,EACA,kBAAgD,oBAAI,IAAI;AAAA,
|
|
4
|
+
"sourcesContent": ["/**\n * Claude Code Subagent Client\n * \n * Uses Claude Code's Task tool to spawn subagents instead of direct API calls\n * This leverages the Claude Code Max plan for unlimited subagent execution\n */\n\nimport { logger } from '../../core/monitoring/logger.js';\nimport { exec } from 'child_process';\nimport { promisify } from 'util';\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport * as os from 'os';\n\nconst execAsync = promisify(exec);\n\nexport interface SubagentRequest {\n type: 'planning' | 'code' | 'testing' | 'linting' | 'review' | 'improve' | 'context' | 'publish';\n task: string;\n context: Record<string, any>;\n systemPrompt?: string;\n files?: string[];\n timeout?: number;\n}\n\nexport interface SubagentResponse {\n success: boolean;\n result: any;\n output?: string;\n error?: string;\n tokens?: number;\n duration: number;\n subagentType: string;\n}\n\n/**\n * Claude Code Subagent Client\n * Spawns subagents using Claude Code's Task tool\n */\nexport class ClaudeCodeSubagentClient {\n private tempDir: string;\n private activeSubagents: Map<string, AbortController> = new Map();\n private mockMode: boolean;\n \n constructor(mockMode: boolean = true) { // Default to mock mode for testing\n this.mockMode = mockMode;\n \n // Create temp directory for subagent communication\n this.tempDir = path.join(os.tmpdir(), 'stackmemory-rlm');\n if (!fs.existsSync(this.tempDir)) {\n fs.mkdirSync(this.tempDir, { recursive: true });\n }\n \n logger.info('Claude Code Subagent Client initialized', {\n tempDir: this.tempDir,\n mockMode: this.mockMode,\n });\n }\n \n /**\n * Execute a subagent task using Claude Code's Task tool\n * This will spawn a new Claude instance with specific instructions\n */\n async executeSubagent(request: SubagentRequest): Promise<SubagentResponse> {\n const startTime = Date.now();\n const subagentId = `${request.type}-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`;\n \n logger.info(`Spawning ${request.type} subagent`, {\n subagentId,\n task: request.task.slice(0, 100),\n mockMode: this.mockMode,\n });\n \n // Return mock responses for testing\n if (this.mockMode) {\n return this.getMockResponse(request, startTime, subagentId);\n }\n \n try {\n // Create subagent prompt based on type\n const prompt = this.buildSubagentPrompt(request);\n \n // Write context to temp file for large contexts\n const contextFile = path.join(this.tempDir, `${subagentId}-context.json`);\n await fs.promises.writeFile(\n contextFile,\n JSON.stringify(request.context, null, 2)\n );\n \n // Create result file path\n const resultFile = path.join(this.tempDir, `${subagentId}-result.json`);\n \n // Build the Task tool command\n // The Task tool will spawn a new Claude instance with the specified prompt\n const taskCommand = this.buildTaskCommand(request, prompt, contextFile, resultFile);\n \n // Execute via Claude Code's Task tool\n const result = await this.executeTaskTool(taskCommand, request.timeout);\n \n // Read result from file\n let subagentResult: any = {};\n if (fs.existsSync(resultFile)) {\n const resultContent = await fs.promises.readFile(resultFile, 'utf-8');\n try {\n subagentResult = JSON.parse(resultContent);\n } catch (e) {\n subagentResult = { rawOutput: resultContent };\n }\n }\n \n // Cleanup temp files\n this.cleanup(subagentId);\n \n return {\n success: true,\n result: subagentResult,\n output: result.stdout,\n duration: Date.now() - startTime,\n subagentType: request.type,\n tokens: this.estimateTokens(prompt + JSON.stringify(subagentResult)),\n };\n \n } catch (error: any) {\n logger.error(`Subagent execution failed: ${request.type}`, { error, subagentId });\n \n return {\n success: false,\n result: null,\n error: error.message,\n duration: Date.now() - startTime,\n subagentType: request.type,\n };\n }\n }\n \n /**\n * Execute multiple subagents in parallel\n */\n async executeParallel(requests: SubagentRequest[]): Promise<SubagentResponse[]> {\n logger.info(`Executing ${requests.length} subagents in parallel`);\n \n const promises = requests.map(request => this.executeSubagent(request));\n const results = await Promise.allSettled(promises);\n \n return results.map((result, index) => {\n if (result.status === 'fulfilled') {\n return result.value;\n } else {\n return {\n success: false,\n result: null,\n error: result.reason?.message || 'Unknown error',\n duration: 0,\n subagentType: requests[index].type,\n };\n }\n });\n }\n \n /**\n * Build subagent prompt based on type\n */\n private buildSubagentPrompt(request: SubagentRequest): string {\n const prompts: Record<string, string> = {\n planning: `You are a Planning Subagent. Your role is to decompose complex tasks into manageable subtasks.\n \n Task: ${request.task}\n \n Instructions:\n 1. Analyze the task and identify all components\n 2. Create a dependency graph of subtasks\n 3. Assign appropriate agent types to each subtask\n 4. Consider parallel execution opportunities\n 5. Include comprehensive testing at each stage\n \n Context is available in the provided file.\n \n Output a JSON structure with the task decomposition.`,\n \n code: `You are a Code Generation Subagent. Your role is to implement high-quality, production-ready code.\n \n Task: ${request.task}\n \n Instructions:\n 1. Write clean, maintainable code\n 2. Follow project conventions (check context)\n 3. Include comprehensive error handling\n 4. Add clear comments for complex logic\n 5. Ensure code is testable\n \n Context and requirements are in the provided file.\n \n Output the implementation code.`,\n \n testing: `You are a Testing Subagent specializing in comprehensive test generation.\n \n Task: ${request.task}\n \n Instructions:\n 1. Generate unit tests for all functions/methods\n 2. Create integration tests for API endpoints\n 3. Add E2E tests for critical user flows\n 4. Include edge cases and error scenarios\n 5. Ensure high code coverage (aim for 100%)\n 6. Validate that all tests pass\n \n Context and code to test are in the provided file.\n \n Output a complete test suite.`,\n \n linting: `You are a Linting Subagent ensuring code quality and standards.\n \n Task: ${request.task}\n \n Instructions:\n 1. Check for syntax errors and type issues\n 2. Verify code formatting and style\n 3. Identify security vulnerabilities\n 4. Find performance anti-patterns\n 5. Detect unused imports and dead code\n 6. Provide specific fixes for each issue\n \n Code to analyze is in the context file.\n \n Output a JSON report with issues and fixes.`,\n \n review: `You are a Code Review Subagent performing thorough multi-stage reviews.\n \n Task: ${request.task}\n \n Instructions:\n 1. Evaluate architecture and design patterns\n 2. Assess code quality and maintainability\n 3. Check performance implications\n 4. Review security considerations\n 5. Verify test coverage adequacy\n 6. Suggest specific improvements with examples\n 7. Rate quality on a 0-1 scale\n \n Code and context are in the provided file.\n \n Output a detailed review with quality score and improvements.`,\n \n improve: `You are an Improvement Subagent enhancing code based on reviews.\n \n Task: ${request.task}\n \n Instructions:\n 1. Implement all suggested improvements\n 2. Refactor for better architecture\n 3. Optimize performance bottlenecks\n 4. Enhance error handling\n 5. Improve code clarity and documentation\n 6. Add missing test cases\n 7. Ensure backward compatibility\n \n Review feedback and code are in the context file.\n \n Output the improved code.`,\n \n context: `You are a Context Retrieval Subagent finding relevant information.\n \n Task: ${request.task}\n \n Instructions:\n 1. Search project codebase for relevant code\n 2. Find similar implementations\n 3. Locate relevant documentation\n 4. Identify dependencies and patterns\n 5. Retrieve best practices\n \n Search parameters are in the context file.\n \n Output relevant context snippets.`,\n \n publish: `You are a Publishing Subagent handling releases and deployments.\n \n Task: ${request.task}\n \n Instructions:\n 1. Prepare package for publishing\n 2. Update version numbers\n 3. Generate changelog\n 4. Create GitHub release\n 5. Publish to NPM if applicable\n 6. Update documentation\n \n Release details are in the context file.\n \n Output the release plan and commands.`,\n };\n \n return request.systemPrompt || prompts[request.type] || prompts.planning;\n }\n \n /**\n * Build Task tool command\n * This creates a command that Claude Code's Task tool can execute\n */\n private buildTaskCommand(\n request: SubagentRequest,\n prompt: string,\n contextFile: string,\n resultFile: string\n ): string {\n // Create a script that the subagent will execute\n const scriptContent = `\n#!/bin/bash\n# Subagent execution script for ${request.type}\n\n# Read context\nCONTEXT=$(cat \"${contextFile}\")\n\n# Execute task based on type\ncase \"${request.type}\" in\n \"testing\")\n # For testing subagent, actually run tests\n echo \"Generating and running tests...\"\n # The subagent will generate test files and run them\n ;;\n \"linting\")\n # For linting subagent, run actual linters\n echo \"Running linters...\"\n npm run lint || true\n ;;\n \"code\")\n # For code generation, create implementation files\n echo \"Generating implementation...\"\n ;;\n *)\n # Default behavior\n echo \"Executing ${request.type} task...\"\n ;;\nesac\n\n# Write result\necho '{\"status\": \"completed\", \"type\": \"${request.type}\"}' > \"${resultFile}\"\n`;\n \n const scriptFile = path.join(this.tempDir, `${request.type}-script.sh`);\n fs.writeFileSync(scriptFile, scriptContent);\n fs.chmodSync(scriptFile, '755');\n \n // Return the command that Task tool will execute\n // In practice, this would trigger Claude Code's Task tool\n return scriptFile;\n }\n \n /**\n * Execute via Task tool (simulated for now)\n * In production, this would use Claude Code's actual Task tool API\n */\n private async executeTaskTool(\n command: string,\n timeout?: number\n ): Promise<{ stdout: string; stderr: string }> {\n try {\n // In production, this would call Claude Code's Task tool\n // For now, we simulate with a subprocess\n const result = await execAsync(command, {\n timeout: timeout || 300000, // 5 minutes default\n maxBuffer: 10 * 1024 * 1024, // 10MB buffer\n });\n \n return result;\n } catch (error: any) {\n if (error.killed || error.signal === 'SIGTERM') {\n throw new Error(`Subagent timeout after ${timeout}ms`);\n }\n throw error;\n }\n }\n \n /**\n * Get mock response for testing\n */\n private async getMockResponse(\n request: SubagentRequest, \n startTime: number, \n subagentId: string\n ): Promise<SubagentResponse> {\n // Simulate some processing time\n await new Promise(resolve => setTimeout(resolve, Math.random() * 20 + 10));\n \n const mockResponses: Record<string, any> = {\n planning: {\n tasks: [\n { id: 'task-1', name: 'Analyze requirements', type: 'analysis' },\n { id: 'task-2', name: 'Design solution', type: 'design' },\n { id: 'task-3', name: 'Implement solution', type: 'implementation' },\n ],\n dependencies: [],\n estimated_time: 300,\n },\n \n code: {\n implementation: `function greetUser(name: string): string {\n if (!name || typeof name !== 'string') {\n throw new Error('Invalid name parameter');\n }\n return \\`Hello, \\${name}!\\`;\n}`,\n files_modified: ['src/greet.ts'],\n lines_added: 6,\n lines_removed: 0,\n },\n \n testing: {\n tests: [\n {\n name: 'greetUser should return greeting',\n code: `test('greetUser should return greeting', () => {\n expect(greetUser('Alice')).toBe('Hello, Alice!');\n});`,\n type: 'unit',\n },\n ],\n coverage: { lines: 100, branches: 100, functions: 100 },\n },\n \n linting: {\n issues: [],\n fixes: [],\n passed: true,\n },\n \n review: {\n quality: 0.85,\n issues: [\n 'Consider adding JSDoc comments',\n 'Could add more edge case tests',\n ],\n suggestions: [\n 'Add documentation for the function',\n 'Consider adding internationalization support',\n 'Add performance tests for large inputs',\n ],\n improvements: [],\n },\n \n improve: {\n improved_code: `/**\n * Greets a user with their name\n * @param name - The name of the user to greet\n * @returns A greeting message\n * @throws {Error} If name is invalid\n */\nfunction greetUser(name: string): string {\n if (!name || typeof name !== 'string') {\n throw new Error('Invalid name parameter: name must be a non-empty string');\n }\n return \\`Hello, \\${name}!\\`;\n}`,\n changes_made: [\n 'Added JSDoc documentation',\n 'Improved error message',\n ],\n },\n \n context: {\n relevant_files: ['src/greet.ts', 'test/greet.test.ts'],\n patterns: ['greeting functions', 'input validation'],\n dependencies: [],\n },\n \n publish: {\n version: '1.0.0',\n changelog: 'Initial release',\n published: false,\n reason: 'Mock mode - no actual publishing',\n },\n };\n \n const result = mockResponses[request.type] || {};\n \n return {\n success: true,\n result,\n output: `Mock ${request.type} subagent completed successfully`,\n duration: Date.now() - startTime,\n subagentType: request.type,\n tokens: this.estimateTokens(JSON.stringify(result)),\n };\n }\n \n /**\n * Estimate token usage\n */\n private estimateTokens(text: string): number {\n // Rough estimation: 1 token \u2248 4 characters\n return Math.ceil(text.length / 4);\n }\n \n /**\n * Cleanup temporary files\n */\n private cleanup(subagentId: string): void {\n const patterns = [\n `${subagentId}-context.json`,\n `${subagentId}-result.json`,\n `${subagentId}-script.sh`,\n ];\n \n for (const pattern of patterns) {\n const filePath = path.join(this.tempDir, pattern);\n if (fs.existsSync(filePath)) {\n try {\n fs.unlinkSync(filePath);\n } catch (e) {\n // Ignore cleanup errors\n }\n }\n }\n }\n \n /**\n * Create a mock Task tool response for development\n * This simulates what Claude Code's Task tool would return\n */\n async mockTaskToolExecution(request: SubagentRequest): Promise<SubagentResponse> {\n const startTime = Date.now();\n \n // Simulate processing delay\n await new Promise(resolve => setTimeout(resolve, 1000 + Math.random() * 2000));\n \n // Generate mock responses based on subagent type\n const mockResponses: Record<string, any> = {\n planning: {\n tasks: [\n { id: '1', type: 'analyze', description: 'Analyze requirements' },\n { id: '2', type: 'implement', description: 'Implement solution' },\n { id: '3', type: 'test', description: 'Test implementation' },\n { id: '4', type: 'review', description: 'Review and improve' },\n ],\n dependencies: { '2': ['1'], '3': ['2'], '4': ['3'] },\n },\n code: {\n implementation: `\nexport class Solution {\n constructor(private config: any) {}\n \n async execute(input: string): Promise<string> {\n // Implementation generated by Code subagent\n return this.process(input);\n }\n \n private process(input: string): string {\n return \\`Processed: \\${input}\\`;\n }\n}`,\n files: ['src/solution.ts'],\n },\n testing: {\n tests: `\ndescribe('Solution', () => {\n it('should process input correctly', () => {\n const solution = new Solution({});\n const result = solution.execute('test');\n expect(result).toBe('Processed: test');\n });\n \n it('should handle edge cases', () => {\n // Edge case tests\n });\n});`,\n coverage: { lines: 95, branches: 88, functions: 100 },\n },\n review: {\n quality: 0.82,\n issues: [\n { severity: 'high', message: 'Missing error handling' },\n { severity: 'medium', message: 'Could improve type safety' },\n ],\n suggestions: [\n 'Add try-catch blocks',\n 'Use stricter TypeScript types',\n 'Add input validation',\n ],\n },\n };\n \n return {\n success: true,\n result: mockResponses[request.type] || { status: 'completed' },\n output: `Mock ${request.type} subagent completed successfully`,\n duration: Date.now() - startTime,\n subagentType: request.type,\n tokens: Math.floor(Math.random() * 5000) + 1000,\n };\n }\n \n /**\n * Get active subagent statistics\n */\n getStats() {\n return {\n activeSubagents: this.activeSubagents.size,\n tempDir: this.tempDir,\n };\n }\n \n /**\n * Cleanup all resources\n */\n async cleanupAll(): Promise<void> {\n // Abort all active subagents\n for (const [id, controller] of this.activeSubagents) {\n controller.abort();\n }\n this.activeSubagents.clear();\n \n // Clean temp directory\n if (fs.existsSync(this.tempDir)) {\n const files = await fs.promises.readdir(this.tempDir);\n for (const file of files) {\n await fs.promises.unlink(path.join(this.tempDir, file));\n }\n }\n \n logger.info('Claude Code Subagent Client cleaned up');\n }\n}"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
function getRailwayConfig() {
|
|
2
|
+
const corsOrigins = process.env.CORS_ORIGINS?.split(",") || [
|
|
3
|
+
"https://claude.ai",
|
|
4
|
+
"https://claude.anthropic.com",
|
|
5
|
+
"http://localhost:3000"
|
|
6
|
+
];
|
|
7
|
+
let redisUrl = process.env.REDIS_URL || "";
|
|
8
|
+
if (!redisUrl && process.env.REDISHOST) {
|
|
9
|
+
const host = process.env.REDISHOST;
|
|
10
|
+
const port = process.env.REDISPORT || "6379";
|
|
11
|
+
const user = process.env.REDISUSER || "default";
|
|
12
|
+
const password = process.env.REDISPASSWORD || "";
|
|
13
|
+
if (password) {
|
|
14
|
+
redisUrl = `redis://${user}:${password}@${host}:${port}`;
|
|
15
|
+
} else {
|
|
16
|
+
redisUrl = `redis://${host}:${port}`;
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
if (!redisUrl && process.env.NODE_ENV !== "production") {
|
|
20
|
+
redisUrl = "redis://localhost:6379";
|
|
21
|
+
}
|
|
22
|
+
return {
|
|
23
|
+
// Server
|
|
24
|
+
port: parseInt(process.env.PORT || "3000"),
|
|
25
|
+
environment: process.env.NODE_ENV || "development",
|
|
26
|
+
// Database - Railway provides this as DATABASE_URL
|
|
27
|
+
databaseUrl: process.env.DATABASE_URL || "postgresql://localhost:5432/stackmemory",
|
|
28
|
+
// Redis
|
|
29
|
+
redisUrl,
|
|
30
|
+
redisHost: process.env.REDISHOST || "localhost",
|
|
31
|
+
redisPort: parseInt(process.env.REDISPORT || "6379"),
|
|
32
|
+
redisUser: process.env.REDISUSER,
|
|
33
|
+
redisPassword: process.env.REDISPASSWORD,
|
|
34
|
+
// Auth
|
|
35
|
+
authMode: process.env.AUTH_MODE || "api_key",
|
|
36
|
+
apiKeySecret: process.env.API_KEY_SECRET || "development-secret",
|
|
37
|
+
jwtSecret: process.env.JWT_SECRET || "development-jwt-secret",
|
|
38
|
+
// Features
|
|
39
|
+
corsOrigins,
|
|
40
|
+
rateLimitEnabled: process.env.RATE_LIMIT_ENABLED === "true",
|
|
41
|
+
rateLimitFree: parseInt(process.env.RATE_LIMIT_FREE || "100"),
|
|
42
|
+
enableWebSocket: process.env.ENABLE_WEBSOCKET !== "false",
|
|
43
|
+
enableAnalytics: process.env.ENABLE_ANALYTICS === "true",
|
|
44
|
+
// Storage mode
|
|
45
|
+
storageMode: process.env.STORAGE_MODE || "hybrid"
|
|
46
|
+
};
|
|
47
|
+
}
|
|
48
|
+
export {
|
|
49
|
+
getRailwayConfig
|
|
50
|
+
};
|
|
51
|
+
//# sourceMappingURL=config.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../src/servers/railway/config.ts"],
|
|
4
|
+
"sourcesContent": ["/**\n * Railway Server Configuration\n * Handles Railway-specific environment variables and service references\n */\n\nexport interface RailwayConfig {\n // Server\n port: number;\n environment: string;\n \n // Database\n databaseUrl: string;\n \n // Redis\n redisUrl: string;\n redisHost: string;\n redisPort: number;\n redisUser?: string;\n redisPassword?: string;\n \n // Auth\n authMode: string;\n apiKeySecret: string;\n jwtSecret: string;\n \n // Features\n corsOrigins: string[];\n rateLimitEnabled: boolean;\n rateLimitFree: number;\n enableWebSocket: boolean;\n enableAnalytics: boolean;\n \n // Storage\n storageMode: 'redis' | 'postgres' | 'hybrid';\n}\n\n/**\n * Get Railway configuration with proper service references\n * Railway provides reference variables like ${{Postgres.DATABASE_URL}}\n */\nexport function getRailwayConfig(): RailwayConfig {\n // Parse CORS origins\n const corsOrigins = process.env.CORS_ORIGINS?.split(',') || [\n 'https://claude.ai',\n 'https://claude.anthropic.com',\n 'http://localhost:3000'\n ];\n \n // Build Redis URL from Railway reference variables\n // Railway typically provides: REDIS_URL or individual components\n let redisUrl = process.env.REDIS_URL || '';\n \n // If no REDIS_URL, try to build from components\n if (!redisUrl && process.env.REDISHOST) {\n const host = process.env.REDISHOST;\n const port = process.env.REDISPORT || '6379';\n const user = process.env.REDISUSER || 'default';\n const password = process.env.REDISPASSWORD || '';\n \n if (password) {\n redisUrl = `redis://${user}:${password}@${host}:${port}`;\n } else {\n redisUrl = `redis://${host}:${port}`;\n }\n }\n \n // Fallback to local Redis if not in production\n if (!redisUrl && process.env.NODE_ENV !== 'production') {\n redisUrl = 'redis://localhost:6379';\n }\n \n return {\n // Server\n port: parseInt(process.env.PORT || '3000'),\n environment: process.env.NODE_ENV || 'development',\n \n // Database - Railway provides this as DATABASE_URL\n databaseUrl: process.env.DATABASE_URL || 'postgresql://localhost:5432/stackmemory',\n \n // Redis\n redisUrl,\n redisHost: process.env.REDISHOST || 'localhost',\n redisPort: parseInt(process.env.REDISPORT || '6379'),\n redisUser: process.env.REDISUSER,\n redisPassword: process.env.REDISPASSWORD,\n \n // Auth\n authMode: process.env.AUTH_MODE || 'api_key',\n apiKeySecret: process.env.API_KEY_SECRET || 'development-secret',\n jwtSecret: process.env.JWT_SECRET || 'development-jwt-secret',\n \n // Features\n corsOrigins,\n rateLimitEnabled: process.env.RATE_LIMIT_ENABLED === 'true',\n rateLimitFree: parseInt(process.env.RATE_LIMIT_FREE || '100'),\n enableWebSocket: process.env.ENABLE_WEBSOCKET !== 'false',\n enableAnalytics: process.env.ENABLE_ANALYTICS === 'true',\n \n // Storage mode\n storageMode: (process.env.STORAGE_MODE as any) || 'hybrid'\n };\n}"],
|
|
5
|
+
"mappings": "AAwCO,SAAS,mBAAkC;AAEhD,QAAM,cAAc,QAAQ,IAAI,cAAc,MAAM,GAAG,KAAK;AAAA,IAC1D;AAAA,IACA;AAAA,IACA;AAAA,EACF;AAIA,MAAI,WAAW,QAAQ,IAAI,aAAa;AAGxC,MAAI,CAAC,YAAY,QAAQ,IAAI,WAAW;AACtC,UAAM,OAAO,QAAQ,IAAI;AACzB,UAAM,OAAO,QAAQ,IAAI,aAAa;AACtC,UAAM,OAAO,QAAQ,IAAI,aAAa;AACtC,UAAM,WAAW,QAAQ,IAAI,iBAAiB;AAE9C,QAAI,UAAU;AACZ,iBAAW,WAAW,IAAI,IAAI,QAAQ,IAAI,IAAI,IAAI,IAAI;AAAA,IACxD,OAAO;AACL,iBAAW,WAAW,IAAI,IAAI,IAAI;AAAA,IACpC;AAAA,EACF;AAGA,MAAI,CAAC,YAAY,QAAQ,IAAI,aAAa,cAAc;AACtD,eAAW;AAAA,EACb;AAEA,SAAO;AAAA;AAAA,IAEL,MAAM,SAAS,QAAQ,IAAI,QAAQ,MAAM;AAAA,IACzC,aAAa,QAAQ,IAAI,YAAY;AAAA;AAAA,IAGrC,aAAa,QAAQ,IAAI,gBAAgB;AAAA;AAAA,IAGzC;AAAA,IACA,WAAW,QAAQ,IAAI,aAAa;AAAA,IACpC,WAAW,SAAS,QAAQ,IAAI,aAAa,MAAM;AAAA,IACnD,WAAW,QAAQ,IAAI;AAAA,IACvB,eAAe,QAAQ,IAAI;AAAA;AAAA,IAG3B,UAAU,QAAQ,IAAI,aAAa;AAAA,IACnC,cAAc,QAAQ,IAAI,kBAAkB;AAAA,IAC5C,WAAW,QAAQ,IAAI,cAAc;AAAA;AAAA,IAGrC;AAAA,IACA,kBAAkB,QAAQ,IAAI,uBAAuB;AAAA,IACrD,eAAe,SAAS,QAAQ,IAAI,mBAAmB,KAAK;AAAA,IAC5D,iBAAiB,QAAQ,IAAI,qBAAqB;AAAA,IAClD,iBAAiB,QAAQ,IAAI,qBAAqB;AAAA;AAAA,IAGlD,aAAc,QAAQ,IAAI,gBAAwB;AAAA,EACpD;AACF;",
|
|
6
|
+
"names": []
|
|
7
|
+
}
|