@stackmemoryai/stackmemory 0.5.33 → 0.5.35

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (111) hide show
  1. package/dist/agents/core/agent-task-manager.js.map +1 -1
  2. package/dist/cli/commands/clear.js +1 -1
  3. package/dist/cli/commands/clear.js.map +1 -1
  4. package/dist/cli/commands/context.js +1 -1
  5. package/dist/cli/commands/context.js.map +1 -1
  6. package/dist/cli/commands/dashboard.js.map +1 -1
  7. package/dist/cli/commands/discovery.js +1 -1
  8. package/dist/cli/commands/discovery.js.map +1 -1
  9. package/dist/cli/commands/handoff.js +1 -1
  10. package/dist/cli/commands/handoff.js.map +1 -1
  11. package/dist/cli/commands/monitor.js +1 -1
  12. package/dist/cli/commands/monitor.js.map +1 -1
  13. package/dist/cli/commands/quality.js +1 -1
  14. package/dist/cli/commands/quality.js.map +1 -1
  15. package/dist/cli/commands/skills.js +1 -1
  16. package/dist/cli/commands/skills.js.map +1 -1
  17. package/dist/cli/commands/workflow.js +1 -1
  18. package/dist/cli/commands/workflow.js.map +1 -1
  19. package/dist/cli/commands/worktree.js +1 -1
  20. package/dist/cli/commands/worktree.js.map +1 -1
  21. package/dist/cli/index.js +1 -1
  22. package/dist/cli/index.js.map +1 -1
  23. package/dist/core/context/auto-context.js.map +1 -1
  24. package/dist/core/context/compaction-handler.js.map +2 -2
  25. package/dist/core/context/context-bridge.js.map +2 -2
  26. package/dist/core/context/dual-stack-manager.js +1 -1
  27. package/dist/core/context/dual-stack-manager.js.map +1 -1
  28. package/dist/core/context/enhanced-rehydration.js.map +1 -1
  29. package/dist/core/context/frame-database.js +43 -10
  30. package/dist/core/context/frame-database.js.map +2 -2
  31. package/dist/core/context/frame-handoff-manager.js.map +1 -1
  32. package/dist/core/context/frame-lifecycle-hooks.js +119 -0
  33. package/dist/core/context/frame-lifecycle-hooks.js.map +7 -0
  34. package/dist/core/context/frame-stack.js +36 -7
  35. package/dist/core/context/frame-stack.js.map +2 -2
  36. package/dist/core/context/incremental-gc.js.map +2 -2
  37. package/dist/core/context/index.js +4 -22
  38. package/dist/core/context/index.js.map +2 -2
  39. package/dist/core/context/refactored-frame-manager.js +170 -37
  40. package/dist/core/context/refactored-frame-manager.js.map +3 -3
  41. package/dist/core/context/shared-context-layer.js.map +1 -1
  42. package/dist/core/context/stack-merge-resolver.js.map +1 -1
  43. package/dist/core/database/database-adapter.js.map +1 -1
  44. package/dist/core/database/paradedb-adapter.js.map +1 -1
  45. package/dist/core/database/query-router.js.map +1 -1
  46. package/dist/core/database/sqlite-adapter.js.map +1 -1
  47. package/dist/core/digest/frame-digest-integration.js.map +1 -1
  48. package/dist/core/digest/hybrid-digest-generator.js.map +1 -1
  49. package/dist/core/digest/types.js.map +1 -1
  50. package/dist/core/errors/index.js +249 -0
  51. package/dist/core/errors/index.js.map +2 -2
  52. package/dist/core/frame/workflow-templates.js.map +2 -2
  53. package/dist/core/merge/conflict-detector.js.map +1 -1
  54. package/dist/core/merge/resolution-engine.js.map +1 -1
  55. package/dist/core/merge/stack-diff.js.map +1 -1
  56. package/dist/core/models/model-router.js +10 -1
  57. package/dist/core/models/model-router.js.map +2 -2
  58. package/dist/core/monitoring/error-handler.js +37 -270
  59. package/dist/core/monitoring/error-handler.js.map +3 -3
  60. package/dist/core/monitoring/session-monitor.js.map +1 -1
  61. package/dist/core/performance/lazy-context-loader.js.map +1 -1
  62. package/dist/core/performance/optimized-frame-context.js.map +1 -1
  63. package/dist/core/retrieval/context-retriever.js.map +1 -1
  64. package/dist/core/retrieval/graph-retrieval.js.map +1 -1
  65. package/dist/core/retrieval/hierarchical-retrieval.js.map +1 -1
  66. package/dist/core/retrieval/llm-context-retrieval.js.map +1 -1
  67. package/dist/core/retrieval/retrieval-benchmarks.js.map +1 -1
  68. package/dist/core/retrieval/summary-generator.js.map +1 -1
  69. package/dist/core/retrieval/types.js.map +1 -1
  70. package/dist/core/storage/chromadb-adapter.js.map +1 -1
  71. package/dist/core/storage/infinite-storage.js.map +1 -1
  72. package/dist/core/storage/two-tier-storage.js.map +1 -1
  73. package/dist/features/tasks/task-aware-context.js.map +1 -1
  74. package/dist/features/web/server/index.js +1 -1
  75. package/dist/features/web/server/index.js.map +1 -1
  76. package/dist/hooks/schemas.js +50 -0
  77. package/dist/hooks/schemas.js.map +2 -2
  78. package/dist/hooks/sms-action-runner.js +47 -1
  79. package/dist/hooks/sms-action-runner.js.map +2 -2
  80. package/dist/hooks/sms-notify.js +63 -1
  81. package/dist/hooks/sms-notify.js.map +2 -2
  82. package/dist/hooks/sms-webhook.js +10 -3
  83. package/dist/hooks/sms-webhook.js.map +2 -2
  84. package/dist/hooks/whatsapp-commands.js +172 -69
  85. package/dist/hooks/whatsapp-commands.js.map +2 -2
  86. package/dist/hooks/whatsapp-sync.js +34 -0
  87. package/dist/hooks/whatsapp-sync.js.map +2 -2
  88. package/dist/index.js +1 -1
  89. package/dist/index.js.map +1 -1
  90. package/dist/integrations/mcp/handlers/context-handlers.js.map +1 -1
  91. package/dist/integrations/mcp/handlers/discovery-handlers.js.map +1 -1
  92. package/dist/integrations/mcp/server.js +1 -1
  93. package/dist/integrations/mcp/server.js.map +1 -1
  94. package/dist/integrations/ralph/bridge/ralph-stackmemory-bridge.js +1 -1
  95. package/dist/integrations/ralph/bridge/ralph-stackmemory-bridge.js.map +1 -1
  96. package/dist/integrations/ralph/context/stackmemory-context-loader.js +1 -1
  97. package/dist/integrations/ralph/context/stackmemory-context-loader.js.map +1 -1
  98. package/dist/integrations/ralph/learning/pattern-learner.js +1 -1
  99. package/dist/integrations/ralph/learning/pattern-learner.js.map +1 -1
  100. package/dist/integrations/ralph/orchestration/multi-loop-orchestrator.js +1 -1
  101. package/dist/integrations/ralph/orchestration/multi-loop-orchestrator.js.map +1 -1
  102. package/dist/integrations/ralph/swarm/swarm-coordinator.js +1 -1
  103. package/dist/integrations/ralph/swarm/swarm-coordinator.js.map +1 -1
  104. package/dist/integrations/ralph/visualization/ralph-debugger.js +1 -1
  105. package/dist/integrations/ralph/visualization/ralph-debugger.js.map +1 -1
  106. package/dist/mcp/stackmemory-mcp-server.js +1 -1
  107. package/dist/mcp/stackmemory-mcp-server.js.map +1 -1
  108. package/dist/skills/claude-skills.js.map +1 -1
  109. package/dist/skills/recursive-agent-orchestrator.js.map +1 -1
  110. package/dist/skills/unified-rlm-orchestrator.js.map +1 -1
  111. package/package.json +1 -1
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/context/dual-stack-manager.ts"],
4
- "sourcesContent": ["/**\n * Dual Stack Manager - STA-99\n * Manages both individual and shared team stacks for collaboration\n */\n\nimport type { Frame, Event, Anchor } from './frame-manager.js';\nimport { FrameManager } from './frame-manager.js';\nimport type { DatabaseAdapter } from '../database/database-adapter.js';\nimport { SQLiteAdapter } from '../database/sqlite-adapter.js';\nimport { logger } from '../monitoring/logger.js';\nimport { ValidationError, DatabaseError, ErrorCode } from '../errors/index.js';\nimport {\n validateInput,\n CreateSharedStackSchema,\n SwitchStackSchema,\n type CreateSharedStackInput,\n type SwitchStackInput,\n} from './validation.js';\nimport { PermissionManager } from './permission-manager.js';\n\nexport interface StackContext {\n stackId: string;\n type: 'individual' | 'shared';\n projectId: string;\n ownerId?: string; // For individual stacks\n teamId?: string; // For shared stacks\n permissions: StackPermissions;\n metadata: Record<string, any>;\n createdAt: Date;\n lastActive: Date;\n}\n\nexport interface StackPermissions {\n canRead: boolean;\n canWrite: boolean;\n canHandoff: boolean;\n canMerge: boolean;\n canAdminister: boolean;\n}\n\nexport interface HandoffRequest {\n requestId: string;\n sourceStackId: string;\n targetStackId: string;\n frameIds: string[];\n requesterId: string;\n targetUserId?: string;\n message?: string;\n status: 'pending' | 'accepted' | 'rejected' | 'expired';\n createdAt: Date;\n expiresAt: Date;\n}\n\nexport interface StackSyncResult {\n success: boolean;\n conflictFrames: string[];\n mergedFrames: string[];\n errors: Array<{\n frameId: string;\n error: string;\n resolution?: 'skipped' | 'merged' | 'manual';\n }>;\n}\n\nexport class DualStackManager {\n private adapter: DatabaseAdapter;\n private individualStack: FrameManager;\n private sharedStacks: Map<string, FrameManager> = new Map();\n private activeContext: StackContext;\n private handoffRequests: Map<string, HandoffRequest> = new Map();\n private permissionManager: PermissionManager;\n\n constructor(\n adapter: DatabaseAdapter,\n projectId: string,\n userId: string,\n defaultTeamId?: string\n ) {\n this.adapter = adapter;\n this.permissionManager = new PermissionManager();\n\n // Initialize individual stack\n // Extract raw database for FrameManager which expects SQLite directly\n const rawDb =\n adapter instanceof SQLiteAdapter ? adapter.getRawDatabase() : null;\n if (!rawDb) {\n throw new DatabaseError(\n 'DualStackManager requires SQLiteAdapter with connected database',\n ErrorCode.DB_CONNECTION_FAILED,\n { adapter: adapter.constructor.name }\n );\n }\n\n this.individualStack = new FrameManager(rawDb, projectId, userId);\n\n // Set default active context to individual stack\n this.activeContext = {\n stackId: `individual-${userId}`,\n type: 'individual',\n projectId,\n ownerId: userId,\n permissions: this.getDefaultIndividualPermissions(),\n metadata: {},\n createdAt: new Date(),\n lastActive: new Date(),\n };\n\n // Set up initial permissions for the user's individual stack\n this.permissionManager.setStackPermissions(\n userId,\n `individual-${userId}`,\n this.getDefaultIndividualPermissions()\n );\n\n this.initializeSchema();\n }\n\n private async initializeSchema(): Promise<void> {\n try {\n // Create stack_contexts table\n await this.adapter.beginTransaction();\n\n const createStackContextsTable = `\n CREATE TABLE IF NOT EXISTS stack_contexts (\n stack_id TEXT PRIMARY KEY,\n type TEXT NOT NULL CHECK (type IN ('individual', 'shared')),\n project_id TEXT NOT NULL,\n owner_id TEXT,\n team_id TEXT,\n permissions TEXT NOT NULL,\n metadata TEXT DEFAULT '{}',\n created_at INTEGER NOT NULL,\n last_active INTEGER NOT NULL,\n CONSTRAINT valid_ownership CHECK (\n (type = 'individual' AND owner_id IS NOT NULL AND team_id IS NULL) OR\n (type = 'shared' AND team_id IS NOT NULL)\n )\n )\n `;\n\n const createHandoffRequestsTable = `\n CREATE TABLE IF NOT EXISTS handoff_requests (\n request_id TEXT PRIMARY KEY,\n source_stack_id TEXT NOT NULL,\n target_stack_id TEXT NOT NULL,\n frame_ids TEXT NOT NULL,\n requester_id TEXT NOT NULL,\n target_user_id TEXT,\n message TEXT,\n status TEXT NOT NULL DEFAULT 'pending' CHECK (status IN ('pending', 'accepted', 'rejected', 'expired')),\n created_at INTEGER NOT NULL,\n expires_at INTEGER NOT NULL,\n FOREIGN KEY (source_stack_id) REFERENCES stack_contexts(stack_id),\n FOREIGN KEY (target_stack_id) REFERENCES stack_contexts(stack_id)\n )\n `;\n\n const createStackSyncLogTable = `\n CREATE TABLE IF NOT EXISTS stack_sync_log (\n sync_id TEXT PRIMARY KEY,\n source_stack_id TEXT NOT NULL,\n target_stack_id TEXT NOT NULL,\n operation TEXT NOT NULL CHECK (operation IN ('handoff', 'merge', 'sync')),\n frame_count INTEGER NOT NULL,\n conflicts TEXT DEFAULT '[]',\n resolution TEXT,\n timestamp INTEGER NOT NULL,\n FOREIGN KEY (source_stack_id) REFERENCES stack_contexts(stack_id),\n FOREIGN KEY (target_stack_id) REFERENCES stack_contexts(stack_id)\n )\n `;\n\n // Execute schema creation using raw SQL\n if (this.adapter.isConnected()) {\n // Note: This is a temporary workaround - proper schema creation would use adapter methods\n (await (this.adapter as any).execute?.(createStackContextsTable)) ||\n this.executeSchemaQuery(createStackContextsTable);\n (await (this.adapter as any).execute?.(createHandoffRequestsTable)) ||\n this.executeSchemaQuery(createHandoffRequestsTable);\n (await (this.adapter as any).execute?.(createStackSyncLogTable)) ||\n this.executeSchemaQuery(createStackSyncLogTable);\n }\n\n await this.adapter.commitTransaction();\n\n logger.info('Dual stack schema initialized successfully');\n } catch (error: unknown) {\n await this.adapter.rollbackTransaction();\n logger.error('Failed to initialize dual stack schema', error);\n throw new DatabaseError(\n 'Schema initialization failed',\n ErrorCode.DB_SCHEMA_ERROR,\n { adapter: this.adapter.constructor.name },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n private async executeSchemaQuery(sql: string): Promise<void> {\n // Fallback for adapters that don't have execute method\n logger.debug(\n 'Using fallback schema creation - implement execute method in adapter'\n );\n\n // Execute using raw SQLite database\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n try {\n rawDb.exec(sql);\n logger.debug('Executed schema query successfully');\n } catch (error: unknown) {\n logger.error('Failed to execute schema query', { sql, error });\n throw error;\n }\n } else {\n throw new DatabaseError(\n 'Cannot execute schema query: raw database not available',\n ErrorCode.DB_CONNECTION_FAILED,\n { operation: 'executeSchemaQuery' }\n );\n }\n }\n\n private getDefaultIndividualPermissions(): StackPermissions {\n return {\n canRead: true,\n canWrite: true,\n canHandoff: true,\n canMerge: true,\n canAdminister: true,\n };\n }\n\n private getSharedStackPermissions(\n role: 'member' | 'lead' | 'admin'\n ): StackPermissions {\n const basePermissions = {\n canRead: true,\n canWrite: true,\n canHandoff: true,\n canMerge: false,\n canAdminister: false,\n };\n\n switch (role) {\n case 'lead':\n return { ...basePermissions, canMerge: true };\n case 'admin':\n return { ...basePermissions, canMerge: true, canAdminister: true };\n default:\n return basePermissions;\n }\n }\n\n /**\n * Switch between individual and shared stacks\n */\n async switchToStack(stackId: string): Promise<void> {\n // Validate input\n const input = validateInput(SwitchStackSchema, { stackId });\n\n try {\n if (input.stackId.startsWith('individual-')) {\n this.activeContext = {\n ...this.activeContext,\n stackId: input.stackId,\n type: 'individual',\n };\n return;\n }\n\n // Load shared stack context\n const stackContext = await this.loadStackContext(input.stackId);\n if (!stackContext) {\n throw new ValidationError(\n `Stack context not found: ${input.stackId}`,\n ErrorCode.STACK_CONTEXT_NOT_FOUND\n );\n }\n\n // Check permission to access the stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n this.activeContext.ownerId || 'unknown',\n 'read',\n 'stack',\n input.stackId,\n stackContext\n )\n );\n\n this.activeContext = stackContext;\n\n // Initialize shared stack manager if not already loaded\n if (!this.sharedStacks.has(input.stackId)) {\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n throw new DatabaseError(\n 'Failed to get raw database for shared stack',\n ErrorCode.DB_CONNECTION_FAILED,\n { stackId: input.stackId, operation: 'switchToStack' }\n );\n }\n\n const sharedStack = new FrameManager(\n rawDb,\n stackContext.projectId,\n input.stackId\n );\n this.sharedStacks.set(input.stackId, sharedStack);\n }\n\n // Update last active timestamp\n await this.updateStackActivity(input.stackId);\n\n logger.info(`Switched to stack: ${input.stackId}`, {\n type: stackContext.type,\n });\n } catch (error: unknown) {\n throw new ValidationError(\n `Failed to switch to stack: ${input.stackId}`,\n ErrorCode.OPERATION_FAILED,\n { stackId: input.stackId },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get the current active stack manager\n */\n getActiveStack(): FrameManager {\n if (this.activeContext.type === 'individual') {\n return this.individualStack;\n }\n\n const sharedStack = this.sharedStacks.get(this.activeContext.stackId);\n if (!sharedStack) {\n throw new DatabaseError(\n `Active shared stack not initialized: ${this.activeContext.stackId}`,\n ErrorCode.INVALID_STATE\n );\n }\n\n return sharedStack;\n }\n\n /**\n * Create a new shared stack for team collaboration\n */\n async createSharedStack(\n teamId: string,\n name: string,\n ownerId: string,\n permissions?: StackPermissions\n ): Promise<string> {\n // Validate input parameters\n const input = validateInput(CreateSharedStackSchema, {\n teamId,\n name,\n ownerId,\n permissions,\n });\n\n // Check permission to create shared stacks\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n input.ownerId,\n 'administer',\n 'stack',\n `shared-${input.teamId}`,\n this.activeContext\n )\n );\n\n const stackId = `shared-${input.teamId}-${Date.now()}`;\n\n const stackContext: StackContext = {\n stackId,\n type: 'shared',\n projectId: this.activeContext.projectId,\n teamId: input.teamId,\n permissions: input.permissions || this.getSharedStackPermissions('admin'),\n metadata: { name: input.name, ownerId: input.ownerId },\n createdAt: new Date(),\n lastActive: new Date(),\n };\n\n try {\n await this.saveStackContext(stackContext);\n\n // Initialize the shared stack manager\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n throw new DatabaseError(\n 'Failed to get raw database for new shared stack',\n ErrorCode.DB_CONNECTION_FAILED,\n { teamId, operation: 'createSharedStack' }\n );\n }\n\n const sharedStack = new FrameManager(\n rawDb,\n stackContext.projectId,\n stackId\n );\n this.sharedStacks.set(stackId, sharedStack);\n\n // Set up permissions for the owner and team\n const stackPermissions = stackContext.permissions;\n this.permissionManager.setStackPermissions(\n input.ownerId,\n stackId,\n stackPermissions\n );\n\n logger.info(`Created shared stack: ${stackId}`, { teamId, name });\n return stackId;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to create shared stack`,\n ErrorCode.OPERATION_FAILED,\n { teamId, name },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Initiate handoff of frames between stacks\n */\n async initiateHandoff(\n targetStackId: string,\n frameIds: string[],\n targetUserId?: string,\n message?: string\n ): Promise<string> {\n // Check permission to perform handoff from current stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n this.activeContext.ownerId || 'unknown',\n 'handoff',\n 'stack',\n this.activeContext.stackId,\n this.activeContext\n )\n );\n\n const requestId = `handoff-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;\n\n const request: HandoffRequest = {\n requestId,\n sourceStackId: this.activeContext.stackId,\n targetStackId,\n frameIds,\n requesterId: this.activeContext.ownerId!,\n targetUserId,\n message,\n status: 'pending',\n createdAt: new Date(),\n expiresAt: new Date(Date.now() + 24 * 60 * 60 * 1000), // 24 hours\n };\n\n try {\n await this.saveHandoffRequest(request);\n this.handoffRequests.set(requestId, request);\n\n logger.info(`Initiated handoff request: ${requestId}`, {\n sourceStack: this.activeContext.stackId,\n targetStack: targetStackId,\n frameCount: frameIds.length,\n });\n\n return requestId;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to initiate handoff`,\n ErrorCode.OPERATION_FAILED,\n { targetStackId, frameIds },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Accept a handoff request and move frames\n */\n async acceptHandoff(requestId: string): Promise<StackSyncResult> {\n logger.debug('acceptHandoff called', { requestId });\n const request = await this.loadHandoffRequest(requestId);\n logger.debug('loadHandoffRequest returned', {\n requestId,\n found: !!request,\n });\n if (!request) {\n logger.error('Handoff request not found', {\n requestId,\n availableRequests: Array.from(this.handoffRequests.keys()),\n });\n throw new DatabaseError(\n `Handoff request not found: ${requestId}`,\n ErrorCode.RESOURCE_NOT_FOUND\n );\n }\n\n if (request.status !== 'pending') {\n throw new DatabaseError(\n `Handoff request is not pending: ${request.status}`,\n ErrorCode.INVALID_STATE\n );\n }\n\n if (request.expiresAt < new Date()) {\n throw new DatabaseError(\n `Handoff request has expired`,\n ErrorCode.OPERATION_EXPIRED\n );\n }\n\n try {\n // Perform the handoff operation\n logger.debug('Starting moveFramesBetweenStacks', { requestId });\n const syncResult = await this.moveFramesBetweenStacks(\n request.sourceStackId,\n request.targetStackId,\n request.frameIds\n );\n logger.debug('moveFramesBetweenStacks completed', {\n requestId,\n success: syncResult.success,\n });\n\n // Update request status\n logger.debug('Updating request status', { requestId });\n request.status = 'accepted';\n logger.debug('Calling saveHandoffRequest', { requestId });\n await this.saveHandoffRequest(request);\n logger.debug('saveHandoffRequest completed', { requestId });\n\n logger.info(`Accepted handoff request: ${requestId}`, {\n frameCount: request.frameIds.length,\n conflicts: syncResult.conflictFrames.length,\n });\n\n return syncResult;\n } catch (error: unknown) {\n logger.error('acceptHandoff caught error', {\n error: error instanceof Error ? error.message : error,\n });\n // Update request status to rejected on failure\n request.status = 'rejected';\n await this.saveHandoffRequest(request);\n\n throw new DatabaseError(\n `Failed to accept handoff`,\n ErrorCode.OPERATION_FAILED,\n { requestId },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Sync frames between individual and shared stacks\n */\n async syncStacks(\n sourceStackId: string,\n targetStackId: string,\n options: {\n frameIds?: string[];\n conflictResolution: 'skip' | 'merge' | 'overwrite';\n dryRun?: boolean;\n }\n ): Promise<StackSyncResult> {\n try {\n const sourceStack = this.getStackManager(sourceStackId);\n const targetStack = this.getStackManager(targetStackId);\n\n // Get frames to sync\n const framesToSync =\n options.frameIds ||\n (await sourceStack.getActiveFrames()).map((f) => f.frame_id);\n\n const result: StackSyncResult = {\n success: true,\n conflictFrames: [],\n mergedFrames: [],\n errors: [],\n };\n\n for (const frameId of framesToSync) {\n try {\n const sourceFrame = await sourceStack.getFrame(frameId);\n if (!sourceFrame) {\n result.errors.push({\n frameId,\n error: 'Source frame not found',\n resolution: 'skipped',\n });\n continue;\n }\n\n const existingFrame = await targetStack.getFrame(frameId);\n\n if (existingFrame) {\n // Handle conflict\n switch (options.conflictResolution) {\n case 'skip':\n result.conflictFrames.push(frameId);\n result.errors.push({\n frameId,\n error: 'Frame already exists',\n resolution: 'skipped',\n });\n continue;\n\n case 'merge':\n if (!options.dryRun) {\n await this.mergeFrames(\n existingFrame,\n sourceFrame,\n targetStack\n );\n }\n result.mergedFrames.push(frameId);\n break;\n\n case 'overwrite':\n if (!options.dryRun) {\n await targetStack.deleteFrame(frameId);\n await this.copyFrame(sourceFrame, targetStack);\n }\n result.mergedFrames.push(frameId);\n break;\n }\n } else {\n // Copy frame to target\n if (!options.dryRun) {\n await this.copyFrame(sourceFrame, targetStack);\n }\n result.mergedFrames.push(frameId);\n }\n } catch (error: unknown) {\n result.errors.push({\n frameId,\n error: error instanceof Error ? error.message : String(error),\n resolution: 'skipped',\n });\n result.success = false;\n }\n }\n\n logger.info(`Stack sync completed`, {\n source: sourceStackId,\n target: targetStackId,\n merged: result.mergedFrames.length,\n conflicts: result.conflictFrames.length,\n errors: result.errors.length,\n });\n\n return result;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Stack sync failed`,\n ErrorCode.OPERATION_FAILED,\n { sourceStackId, targetStackId },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n getStackManager(stackId: string): FrameManager {\n logger.debug('getStackManager called', {\n stackId,\n availableStacks: Array.from(this.sharedStacks.keys()),\n });\n\n if (stackId.startsWith('individual-')) {\n logger.debug('Returning individual stack', { stackId });\n return this.individualStack;\n }\n\n const sharedStack = this.sharedStacks.get(stackId);\n if (!sharedStack) {\n logger.error('Stack manager not found', {\n stackId,\n availableSharedStacks: Array.from(this.sharedStacks.keys()),\n message: 'getStackManager could not find shared stack',\n });\n throw new DatabaseError(\n `Stack manager not found: ${stackId}`,\n ErrorCode.RESOURCE_NOT_FOUND\n );\n }\n\n logger.debug('Returning shared stack', { stackId });\n return sharedStack;\n }\n\n private async moveFramesBetweenStacks(\n sourceStackId: string,\n targetStackId: string,\n frameIds: string[]\n ): Promise<StackSyncResult> {\n const syncResult = await this.syncStacks(sourceStackId, targetStackId, {\n frameIds,\n conflictResolution: 'merge',\n });\n\n // Remove frames from source stack after successful sync\n if (syncResult.success && syncResult.errors.length === 0) {\n const sourceStack = this.getStackManager(sourceStackId);\n for (const frameId of frameIds) {\n try {\n sourceStack.deleteFrame(frameId);\n logger.debug('Deleted frame from source stack', {\n frameId,\n sourceStackId,\n });\n } catch (error: unknown) {\n logger.warn('Failed to delete frame from source stack', {\n frameId,\n error,\n });\n }\n }\n logger.debug('Completed frame cleanup from source stack', {\n frameIds: frameIds.length,\n });\n }\n\n return syncResult;\n }\n\n private async copyFrame(\n frame: Frame,\n targetStack: FrameManager\n ): Promise<void> {\n // Create frame in target stack\n await targetStack.createFrame({\n type: frame.type as any,\n name: frame.name,\n inputs: frame.inputs,\n });\n\n // Copy events\n const events = await this.individualStack.getFrameEvents(frame.frame_id);\n for (const event of events) {\n await targetStack.addEvent(frame.frame_id, {\n type: event.type as any,\n text: event.text,\n metadata: event.metadata,\n });\n }\n\n // Copy anchors\n const anchors = await this.individualStack.getFrameAnchors(frame.frame_id);\n for (const anchor of anchors) {\n await targetStack.addAnchor(frame.frame_id, {\n type: anchor.type as any,\n text: anchor.text,\n priority: anchor.priority,\n metadata: anchor.metadata,\n });\n }\n }\n\n private async mergeFrames(\n existingFrame: Frame,\n sourceFrame: Frame,\n targetStack: FrameManager\n ): Promise<void> {\n // Simple merge strategy - append events and anchors\n const sourceEvents = await this.individualStack.getFrameEvents(\n sourceFrame.frame_id\n );\n for (const event of sourceEvents) {\n await targetStack.addEvent(existingFrame.frame_id, {\n type: event.type as any,\n text: event.text,\n metadata: { ...event.metadata, merged: true },\n });\n }\n\n const sourceAnchors = await this.individualStack.getFrameAnchors(\n sourceFrame.frame_id\n );\n for (const anchor of sourceAnchors) {\n await targetStack.addAnchor(existingFrame.frame_id, {\n type: anchor.type as any,\n text: anchor.text,\n priority: anchor.priority,\n metadata: { ...anchor.metadata, merged: true },\n });\n }\n }\n\n private async loadStackContext(\n stackId: string\n ): Promise<StackContext | null> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n return null;\n }\n\n const query = rawDb.prepare(`\n SELECT stack_id, type, project_id, owner_id, team_id, permissions, metadata, created_at, last_active\n FROM stack_contexts \n WHERE stack_id = ?\n `);\n\n const row = query.get(stackId) as any;\n if (!row) {\n return null;\n }\n\n return {\n stackId: row.stack_id,\n type: row.type,\n projectId: row.project_id,\n ownerId: row.owner_id,\n teamId: row.team_id,\n permissions: JSON.parse(row.permissions),\n metadata: JSON.parse(row.metadata || '{}'),\n createdAt: new Date(row.created_at),\n lastActive: new Date(row.last_active),\n };\n } catch (error: unknown) {\n logger.error('Failed to load stack context', { stackId, error });\n return null;\n }\n }\n\n private async saveStackContext(context: StackContext): Promise<void> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n throw new DatabaseError(\n 'SQLite database not available for stack context save',\n ErrorCode.DB_CONNECTION_FAILED,\n { stackId: context.stackId, operation: 'saveStackContext' }\n );\n }\n\n const query = rawDb.prepare(`\n INSERT OR REPLACE INTO stack_contexts \n (stack_id, type, project_id, owner_id, team_id, permissions, metadata, created_at, last_active)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)\n `);\n\n query.run(\n context.stackId,\n context.type,\n context.projectId,\n context.ownerId || null,\n context.teamId || null,\n JSON.stringify(context.permissions),\n JSON.stringify(context.metadata || {}),\n context.createdAt.getTime(),\n context.lastActive.getTime()\n );\n\n logger.debug('Saved stack context', { stackId: context.stackId });\n } catch (error: unknown) {\n logger.error('Failed to save stack context', {\n stackId: context.stackId,\n error,\n });\n throw error;\n }\n }\n\n private async updateStackActivity(stackId: string): Promise<void> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n logger.warn('SQLite database not available for activity update');\n return;\n }\n\n const query = rawDb.prepare(`\n UPDATE stack_contexts \n SET last_active = ?\n WHERE stack_id = ?\n `);\n\n query.run(Date.now(), stackId);\n logger.debug('Updated stack activity', { stackId });\n } catch (error: unknown) {\n logger.error('Failed to update stack activity', { stackId, error });\n // Don't throw - activity updates are not critical\n }\n }\n\n private async loadHandoffRequest(\n requestId: string\n ): Promise<HandoffRequest | null> {\n // Try in-memory first for fast access\n const memoryRequest = this.handoffRequests.get(requestId);\n if (memoryRequest) {\n return memoryRequest;\n }\n\n // Try loading from database\n try {\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n const query = rawDb.prepare(`\n SELECT * FROM handoff_requests WHERE request_id = ?\n `);\n\n const row = query.get(requestId) as any;\n if (row) {\n const request: HandoffRequest = {\n requestId: row.request_id,\n sourceStackId: row.source_stack_id,\n targetStackId: row.target_stack_id,\n frameIds: JSON.parse(row.frame_ids),\n status: row.status,\n createdAt: new Date(row.created_at),\n expiresAt: new Date(row.expires_at),\n targetUserId: row.target_user_id,\n message: row.message,\n };\n\n // Cache in memory for future access\n this.handoffRequests.set(requestId, request);\n return request;\n }\n }\n } catch (error: unknown) {\n logger.error('Failed to load handoff request from database', {\n requestId,\n error,\n });\n }\n\n return null;\n }\n\n private async saveHandoffRequest(request: HandoffRequest): Promise<void> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n const query = rawDb.prepare(`\n INSERT OR REPLACE INTO handoff_requests \n (request_id, source_stack_id, target_stack_id, frame_ids, status, created_at, expires_at, target_user_id, message)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)\n `);\n\n query.run(\n request.requestId,\n request.sourceStackId,\n request.targetStackId,\n JSON.stringify(request.frameIds),\n request.status,\n request.createdAt.getTime(),\n request.expiresAt.getTime(),\n request.targetUserId || null,\n request.message || null\n );\n\n logger.debug('Saved handoff request to database', {\n requestId: request.requestId,\n });\n }\n\n // Also keep in-memory for fast access\n this.handoffRequests.set(request.requestId, request);\n } catch (error: unknown) {\n logger.error('Failed to save handoff request', {\n requestId: request.requestId,\n error,\n });\n // Fallback to in-memory only\n this.handoffRequests.set(request.requestId, request);\n }\n }\n\n /**\n * Get list of available stacks for the current user\n */\n async getAvailableStacks(): Promise<StackContext[]> {\n try {\n const stacks: StackContext[] = [];\n\n // Always include current individual stack context\n stacks.push(this.activeContext);\n\n // Query database for all shared stacks the user has access to\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n const query = rawDb.prepare(`\n SELECT stack_id, type, project_id, owner_id, team_id, permissions, metadata, created_at, last_active\n FROM stack_contexts \n WHERE type = 'shared' AND project_id = ?\n `);\n\n const rows = query.all(this.activeContext.projectId) as any[];\n\n for (const row of rows) {\n const context: StackContext = {\n stackId: row.stack_id,\n type: row.type,\n projectId: row.project_id,\n ownerId: row.owner_id,\n teamId: row.team_id,\n permissions: JSON.parse(row.permissions),\n metadata: JSON.parse(row.metadata || '{}'),\n createdAt: new Date(row.created_at),\n lastActive: new Date(row.last_active),\n };\n\n // Check if user has permission to access this stack\n try {\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n this.activeContext.ownerId || 'unknown',\n 'read',\n 'stack',\n context.stackId,\n context\n )\n );\n stacks.push(context);\n } catch (permissionError: unknown) {\n // Skip stacks user doesn't have access to\n logger.debug('User lacks access to stack', {\n stackId: context.stackId,\n userId: this.activeContext.ownerId,\n });\n }\n }\n }\n\n return stacks;\n } catch (error: unknown) {\n logger.error('Failed to get available stacks', error);\n // Return at least the current individual stack\n return [this.activeContext];\n }\n }\n\n /**\n * Get pending handoff requests for the current user\n */\n async getPendingHandoffRequests(): Promise<HandoffRequest[]> {\n return Array.from(this.handoffRequests.values()).filter(\n (request) =>\n request.status === 'pending' && request.expiresAt > new Date()\n );\n }\n\n /**\n * Get current stack context\n */\n getCurrentContext(): StackContext {\n return { ...this.activeContext };\n }\n\n /**\n * Get permission manager for external access\n */\n getPermissionManager(): PermissionManager {\n return this.permissionManager;\n }\n\n /**\n * Add user to shared stack with specific permissions\n */\n async addUserToSharedStack(\n stackId: string,\n userId: string,\n permissions: StackPermissions,\n requesterId: string\n ): Promise<void> {\n // Check if requester has admin permissions on the stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n requesterId,\n 'administer',\n 'stack',\n stackId\n )\n );\n\n // Grant permissions to the new user\n this.permissionManager.setStackPermissions(userId, stackId, permissions);\n\n logger.info(`Added user to shared stack`, {\n stackId,\n userId,\n permissions,\n requesterId,\n });\n }\n\n /**\n * Remove user from shared stack\n */\n async removeUserFromSharedStack(\n stackId: string,\n userId: string,\n requesterId: string\n ): Promise<void> {\n // Check if requester has admin permissions on the stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n requesterId,\n 'administer',\n 'stack',\n stackId\n )\n );\n\n // Remove user's permissions\n const userPermissions = this.permissionManager.getUserPermissions(userId);\n userPermissions.delete(stackId);\n\n logger.info(`Removed user from shared stack`, {\n stackId,\n userId,\n requesterId,\n });\n }\n}\n"],
4
+ "sourcesContent": ["/**\n * Dual Stack Manager - STA-99\n * Manages both individual and shared team stacks for collaboration\n */\n\nimport type { Frame, Event, Anchor } from './frame-types.js';\nimport { FrameManager } from './index.js';\nimport type { DatabaseAdapter } from '../database/database-adapter.js';\nimport { SQLiteAdapter } from '../database/sqlite-adapter.js';\nimport { logger } from '../monitoring/logger.js';\nimport { ValidationError, DatabaseError, ErrorCode } from '../errors/index.js';\nimport {\n validateInput,\n CreateSharedStackSchema,\n SwitchStackSchema,\n type CreateSharedStackInput,\n type SwitchStackInput,\n} from './validation.js';\nimport { PermissionManager } from './permission-manager.js';\n\nexport interface StackContext {\n stackId: string;\n type: 'individual' | 'shared';\n projectId: string;\n ownerId?: string; // For individual stacks\n teamId?: string; // For shared stacks\n permissions: StackPermissions;\n metadata: Record<string, any>;\n createdAt: Date;\n lastActive: Date;\n}\n\nexport interface StackPermissions {\n canRead: boolean;\n canWrite: boolean;\n canHandoff: boolean;\n canMerge: boolean;\n canAdminister: boolean;\n}\n\nexport interface HandoffRequest {\n requestId: string;\n sourceStackId: string;\n targetStackId: string;\n frameIds: string[];\n requesterId: string;\n targetUserId?: string;\n message?: string;\n status: 'pending' | 'accepted' | 'rejected' | 'expired';\n createdAt: Date;\n expiresAt: Date;\n}\n\nexport interface StackSyncResult {\n success: boolean;\n conflictFrames: string[];\n mergedFrames: string[];\n errors: Array<{\n frameId: string;\n error: string;\n resolution?: 'skipped' | 'merged' | 'manual';\n }>;\n}\n\nexport class DualStackManager {\n private adapter: DatabaseAdapter;\n private individualStack: FrameManager;\n private sharedStacks: Map<string, FrameManager> = new Map();\n private activeContext: StackContext;\n private handoffRequests: Map<string, HandoffRequest> = new Map();\n private permissionManager: PermissionManager;\n\n constructor(\n adapter: DatabaseAdapter,\n projectId: string,\n userId: string,\n defaultTeamId?: string\n ) {\n this.adapter = adapter;\n this.permissionManager = new PermissionManager();\n\n // Initialize individual stack\n // Extract raw database for FrameManager which expects SQLite directly\n const rawDb =\n adapter instanceof SQLiteAdapter ? adapter.getRawDatabase() : null;\n if (!rawDb) {\n throw new DatabaseError(\n 'DualStackManager requires SQLiteAdapter with connected database',\n ErrorCode.DB_CONNECTION_FAILED,\n { adapter: adapter.constructor.name }\n );\n }\n\n this.individualStack = new FrameManager(rawDb, projectId, userId);\n\n // Set default active context to individual stack\n this.activeContext = {\n stackId: `individual-${userId}`,\n type: 'individual',\n projectId,\n ownerId: userId,\n permissions: this.getDefaultIndividualPermissions(),\n metadata: {},\n createdAt: new Date(),\n lastActive: new Date(),\n };\n\n // Set up initial permissions for the user's individual stack\n this.permissionManager.setStackPermissions(\n userId,\n `individual-${userId}`,\n this.getDefaultIndividualPermissions()\n );\n\n this.initializeSchema();\n }\n\n private async initializeSchema(): Promise<void> {\n try {\n // Create stack_contexts table\n await this.adapter.beginTransaction();\n\n const createStackContextsTable = `\n CREATE TABLE IF NOT EXISTS stack_contexts (\n stack_id TEXT PRIMARY KEY,\n type TEXT NOT NULL CHECK (type IN ('individual', 'shared')),\n project_id TEXT NOT NULL,\n owner_id TEXT,\n team_id TEXT,\n permissions TEXT NOT NULL,\n metadata TEXT DEFAULT '{}',\n created_at INTEGER NOT NULL,\n last_active INTEGER NOT NULL,\n CONSTRAINT valid_ownership CHECK (\n (type = 'individual' AND owner_id IS NOT NULL AND team_id IS NULL) OR\n (type = 'shared' AND team_id IS NOT NULL)\n )\n )\n `;\n\n const createHandoffRequestsTable = `\n CREATE TABLE IF NOT EXISTS handoff_requests (\n request_id TEXT PRIMARY KEY,\n source_stack_id TEXT NOT NULL,\n target_stack_id TEXT NOT NULL,\n frame_ids TEXT NOT NULL,\n requester_id TEXT NOT NULL,\n target_user_id TEXT,\n message TEXT,\n status TEXT NOT NULL DEFAULT 'pending' CHECK (status IN ('pending', 'accepted', 'rejected', 'expired')),\n created_at INTEGER NOT NULL,\n expires_at INTEGER NOT NULL,\n FOREIGN KEY (source_stack_id) REFERENCES stack_contexts(stack_id),\n FOREIGN KEY (target_stack_id) REFERENCES stack_contexts(stack_id)\n )\n `;\n\n const createStackSyncLogTable = `\n CREATE TABLE IF NOT EXISTS stack_sync_log (\n sync_id TEXT PRIMARY KEY,\n source_stack_id TEXT NOT NULL,\n target_stack_id TEXT NOT NULL,\n operation TEXT NOT NULL CHECK (operation IN ('handoff', 'merge', 'sync')),\n frame_count INTEGER NOT NULL,\n conflicts TEXT DEFAULT '[]',\n resolution TEXT,\n timestamp INTEGER NOT NULL,\n FOREIGN KEY (source_stack_id) REFERENCES stack_contexts(stack_id),\n FOREIGN KEY (target_stack_id) REFERENCES stack_contexts(stack_id)\n )\n `;\n\n // Execute schema creation using raw SQL\n if (this.adapter.isConnected()) {\n // Note: This is a temporary workaround - proper schema creation would use adapter methods\n (await (this.adapter as any).execute?.(createStackContextsTable)) ||\n this.executeSchemaQuery(createStackContextsTable);\n (await (this.adapter as any).execute?.(createHandoffRequestsTable)) ||\n this.executeSchemaQuery(createHandoffRequestsTable);\n (await (this.adapter as any).execute?.(createStackSyncLogTable)) ||\n this.executeSchemaQuery(createStackSyncLogTable);\n }\n\n await this.adapter.commitTransaction();\n\n logger.info('Dual stack schema initialized successfully');\n } catch (error: unknown) {\n await this.adapter.rollbackTransaction();\n logger.error('Failed to initialize dual stack schema', error);\n throw new DatabaseError(\n 'Schema initialization failed',\n ErrorCode.DB_SCHEMA_ERROR,\n { adapter: this.adapter.constructor.name },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n private async executeSchemaQuery(sql: string): Promise<void> {\n // Fallback for adapters that don't have execute method\n logger.debug(\n 'Using fallback schema creation - implement execute method in adapter'\n );\n\n // Execute using raw SQLite database\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n try {\n rawDb.exec(sql);\n logger.debug('Executed schema query successfully');\n } catch (error: unknown) {\n logger.error('Failed to execute schema query', { sql, error });\n throw error;\n }\n } else {\n throw new DatabaseError(\n 'Cannot execute schema query: raw database not available',\n ErrorCode.DB_CONNECTION_FAILED,\n { operation: 'executeSchemaQuery' }\n );\n }\n }\n\n private getDefaultIndividualPermissions(): StackPermissions {\n return {\n canRead: true,\n canWrite: true,\n canHandoff: true,\n canMerge: true,\n canAdminister: true,\n };\n }\n\n private getSharedStackPermissions(\n role: 'member' | 'lead' | 'admin'\n ): StackPermissions {\n const basePermissions = {\n canRead: true,\n canWrite: true,\n canHandoff: true,\n canMerge: false,\n canAdminister: false,\n };\n\n switch (role) {\n case 'lead':\n return { ...basePermissions, canMerge: true };\n case 'admin':\n return { ...basePermissions, canMerge: true, canAdminister: true };\n default:\n return basePermissions;\n }\n }\n\n /**\n * Switch between individual and shared stacks\n */\n async switchToStack(stackId: string): Promise<void> {\n // Validate input\n const input = validateInput(SwitchStackSchema, { stackId });\n\n try {\n if (input.stackId.startsWith('individual-')) {\n this.activeContext = {\n ...this.activeContext,\n stackId: input.stackId,\n type: 'individual',\n };\n return;\n }\n\n // Load shared stack context\n const stackContext = await this.loadStackContext(input.stackId);\n if (!stackContext) {\n throw new ValidationError(\n `Stack context not found: ${input.stackId}`,\n ErrorCode.STACK_CONTEXT_NOT_FOUND\n );\n }\n\n // Check permission to access the stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n this.activeContext.ownerId || 'unknown',\n 'read',\n 'stack',\n input.stackId,\n stackContext\n )\n );\n\n this.activeContext = stackContext;\n\n // Initialize shared stack manager if not already loaded\n if (!this.sharedStacks.has(input.stackId)) {\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n throw new DatabaseError(\n 'Failed to get raw database for shared stack',\n ErrorCode.DB_CONNECTION_FAILED,\n { stackId: input.stackId, operation: 'switchToStack' }\n );\n }\n\n const sharedStack = new FrameManager(\n rawDb,\n stackContext.projectId,\n input.stackId\n );\n this.sharedStacks.set(input.stackId, sharedStack);\n }\n\n // Update last active timestamp\n await this.updateStackActivity(input.stackId);\n\n logger.info(`Switched to stack: ${input.stackId}`, {\n type: stackContext.type,\n });\n } catch (error: unknown) {\n throw new ValidationError(\n `Failed to switch to stack: ${input.stackId}`,\n ErrorCode.OPERATION_FAILED,\n { stackId: input.stackId },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get the current active stack manager\n */\n getActiveStack(): FrameManager {\n if (this.activeContext.type === 'individual') {\n return this.individualStack;\n }\n\n const sharedStack = this.sharedStacks.get(this.activeContext.stackId);\n if (!sharedStack) {\n throw new DatabaseError(\n `Active shared stack not initialized: ${this.activeContext.stackId}`,\n ErrorCode.INVALID_STATE\n );\n }\n\n return sharedStack;\n }\n\n /**\n * Create a new shared stack for team collaboration\n */\n async createSharedStack(\n teamId: string,\n name: string,\n ownerId: string,\n permissions?: StackPermissions\n ): Promise<string> {\n // Validate input parameters\n const input = validateInput(CreateSharedStackSchema, {\n teamId,\n name,\n ownerId,\n permissions,\n });\n\n // Check permission to create shared stacks\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n input.ownerId,\n 'administer',\n 'stack',\n `shared-${input.teamId}`,\n this.activeContext\n )\n );\n\n const stackId = `shared-${input.teamId}-${Date.now()}`;\n\n const stackContext: StackContext = {\n stackId,\n type: 'shared',\n projectId: this.activeContext.projectId,\n teamId: input.teamId,\n permissions: input.permissions || this.getSharedStackPermissions('admin'),\n metadata: { name: input.name, ownerId: input.ownerId },\n createdAt: new Date(),\n lastActive: new Date(),\n };\n\n try {\n await this.saveStackContext(stackContext);\n\n // Initialize the shared stack manager\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n throw new DatabaseError(\n 'Failed to get raw database for new shared stack',\n ErrorCode.DB_CONNECTION_FAILED,\n { teamId, operation: 'createSharedStack' }\n );\n }\n\n const sharedStack = new FrameManager(\n rawDb,\n stackContext.projectId,\n stackId\n );\n this.sharedStacks.set(stackId, sharedStack);\n\n // Set up permissions for the owner and team\n const stackPermissions = stackContext.permissions;\n this.permissionManager.setStackPermissions(\n input.ownerId,\n stackId,\n stackPermissions\n );\n\n logger.info(`Created shared stack: ${stackId}`, { teamId, name });\n return stackId;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to create shared stack`,\n ErrorCode.OPERATION_FAILED,\n { teamId, name },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Initiate handoff of frames between stacks\n */\n async initiateHandoff(\n targetStackId: string,\n frameIds: string[],\n targetUserId?: string,\n message?: string\n ): Promise<string> {\n // Check permission to perform handoff from current stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n this.activeContext.ownerId || 'unknown',\n 'handoff',\n 'stack',\n this.activeContext.stackId,\n this.activeContext\n )\n );\n\n const requestId = `handoff-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;\n\n const request: HandoffRequest = {\n requestId,\n sourceStackId: this.activeContext.stackId,\n targetStackId,\n frameIds,\n requesterId: this.activeContext.ownerId!,\n targetUserId,\n message,\n status: 'pending',\n createdAt: new Date(),\n expiresAt: new Date(Date.now() + 24 * 60 * 60 * 1000), // 24 hours\n };\n\n try {\n await this.saveHandoffRequest(request);\n this.handoffRequests.set(requestId, request);\n\n logger.info(`Initiated handoff request: ${requestId}`, {\n sourceStack: this.activeContext.stackId,\n targetStack: targetStackId,\n frameCount: frameIds.length,\n });\n\n return requestId;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to initiate handoff`,\n ErrorCode.OPERATION_FAILED,\n { targetStackId, frameIds },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Accept a handoff request and move frames\n */\n async acceptHandoff(requestId: string): Promise<StackSyncResult> {\n logger.debug('acceptHandoff called', { requestId });\n const request = await this.loadHandoffRequest(requestId);\n logger.debug('loadHandoffRequest returned', {\n requestId,\n found: !!request,\n });\n if (!request) {\n logger.error('Handoff request not found', {\n requestId,\n availableRequests: Array.from(this.handoffRequests.keys()),\n });\n throw new DatabaseError(\n `Handoff request not found: ${requestId}`,\n ErrorCode.RESOURCE_NOT_FOUND\n );\n }\n\n if (request.status !== 'pending') {\n throw new DatabaseError(\n `Handoff request is not pending: ${request.status}`,\n ErrorCode.INVALID_STATE\n );\n }\n\n if (request.expiresAt < new Date()) {\n throw new DatabaseError(\n `Handoff request has expired`,\n ErrorCode.OPERATION_EXPIRED\n );\n }\n\n try {\n // Perform the handoff operation\n logger.debug('Starting moveFramesBetweenStacks', { requestId });\n const syncResult = await this.moveFramesBetweenStacks(\n request.sourceStackId,\n request.targetStackId,\n request.frameIds\n );\n logger.debug('moveFramesBetweenStacks completed', {\n requestId,\n success: syncResult.success,\n });\n\n // Update request status\n logger.debug('Updating request status', { requestId });\n request.status = 'accepted';\n logger.debug('Calling saveHandoffRequest', { requestId });\n await this.saveHandoffRequest(request);\n logger.debug('saveHandoffRequest completed', { requestId });\n\n logger.info(`Accepted handoff request: ${requestId}`, {\n frameCount: request.frameIds.length,\n conflicts: syncResult.conflictFrames.length,\n });\n\n return syncResult;\n } catch (error: unknown) {\n logger.error('acceptHandoff caught error', {\n error: error instanceof Error ? error.message : error,\n });\n // Update request status to rejected on failure\n request.status = 'rejected';\n await this.saveHandoffRequest(request);\n\n throw new DatabaseError(\n `Failed to accept handoff`,\n ErrorCode.OPERATION_FAILED,\n { requestId },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Sync frames between individual and shared stacks\n */\n async syncStacks(\n sourceStackId: string,\n targetStackId: string,\n options: {\n frameIds?: string[];\n conflictResolution: 'skip' | 'merge' | 'overwrite';\n dryRun?: boolean;\n }\n ): Promise<StackSyncResult> {\n try {\n const sourceStack = this.getStackManager(sourceStackId);\n const targetStack = this.getStackManager(targetStackId);\n\n // Get frames to sync\n const framesToSync =\n options.frameIds ||\n (await sourceStack.getActiveFrames()).map((f) => f.frame_id);\n\n const result: StackSyncResult = {\n success: true,\n conflictFrames: [],\n mergedFrames: [],\n errors: [],\n };\n\n for (const frameId of framesToSync) {\n try {\n const sourceFrame = await sourceStack.getFrame(frameId);\n if (!sourceFrame) {\n result.errors.push({\n frameId,\n error: 'Source frame not found',\n resolution: 'skipped',\n });\n continue;\n }\n\n const existingFrame = await targetStack.getFrame(frameId);\n\n if (existingFrame) {\n // Handle conflict\n switch (options.conflictResolution) {\n case 'skip':\n result.conflictFrames.push(frameId);\n result.errors.push({\n frameId,\n error: 'Frame already exists',\n resolution: 'skipped',\n });\n continue;\n\n case 'merge':\n if (!options.dryRun) {\n await this.mergeFrames(\n existingFrame,\n sourceFrame,\n targetStack\n );\n }\n result.mergedFrames.push(frameId);\n break;\n\n case 'overwrite':\n if (!options.dryRun) {\n await targetStack.deleteFrame(frameId);\n await this.copyFrame(sourceFrame, targetStack);\n }\n result.mergedFrames.push(frameId);\n break;\n }\n } else {\n // Copy frame to target\n if (!options.dryRun) {\n await this.copyFrame(sourceFrame, targetStack);\n }\n result.mergedFrames.push(frameId);\n }\n } catch (error: unknown) {\n result.errors.push({\n frameId,\n error: error instanceof Error ? error.message : String(error),\n resolution: 'skipped',\n });\n result.success = false;\n }\n }\n\n logger.info(`Stack sync completed`, {\n source: sourceStackId,\n target: targetStackId,\n merged: result.mergedFrames.length,\n conflicts: result.conflictFrames.length,\n errors: result.errors.length,\n });\n\n return result;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Stack sync failed`,\n ErrorCode.OPERATION_FAILED,\n { sourceStackId, targetStackId },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n getStackManager(stackId: string): FrameManager {\n logger.debug('getStackManager called', {\n stackId,\n availableStacks: Array.from(this.sharedStacks.keys()),\n });\n\n if (stackId.startsWith('individual-')) {\n logger.debug('Returning individual stack', { stackId });\n return this.individualStack;\n }\n\n const sharedStack = this.sharedStacks.get(stackId);\n if (!sharedStack) {\n logger.error('Stack manager not found', {\n stackId,\n availableSharedStacks: Array.from(this.sharedStacks.keys()),\n message: 'getStackManager could not find shared stack',\n });\n throw new DatabaseError(\n `Stack manager not found: ${stackId}`,\n ErrorCode.RESOURCE_NOT_FOUND\n );\n }\n\n logger.debug('Returning shared stack', { stackId });\n return sharedStack;\n }\n\n private async moveFramesBetweenStacks(\n sourceStackId: string,\n targetStackId: string,\n frameIds: string[]\n ): Promise<StackSyncResult> {\n const syncResult = await this.syncStacks(sourceStackId, targetStackId, {\n frameIds,\n conflictResolution: 'merge',\n });\n\n // Remove frames from source stack after successful sync\n if (syncResult.success && syncResult.errors.length === 0) {\n const sourceStack = this.getStackManager(sourceStackId);\n for (const frameId of frameIds) {\n try {\n sourceStack.deleteFrame(frameId);\n logger.debug('Deleted frame from source stack', {\n frameId,\n sourceStackId,\n });\n } catch (error: unknown) {\n logger.warn('Failed to delete frame from source stack', {\n frameId,\n error,\n });\n }\n }\n logger.debug('Completed frame cleanup from source stack', {\n frameIds: frameIds.length,\n });\n }\n\n return syncResult;\n }\n\n private async copyFrame(\n frame: Frame,\n targetStack: FrameManager\n ): Promise<void> {\n // Create frame in target stack\n await targetStack.createFrame({\n type: frame.type as any,\n name: frame.name,\n inputs: frame.inputs,\n });\n\n // Copy events\n const events = await this.individualStack.getFrameEvents(frame.frame_id);\n for (const event of events) {\n await targetStack.addEvent(frame.frame_id, {\n type: event.type as any,\n text: event.text,\n metadata: event.metadata,\n });\n }\n\n // Copy anchors\n const anchors = await this.individualStack.getFrameAnchors(frame.frame_id);\n for (const anchor of anchors) {\n await targetStack.addAnchor(frame.frame_id, {\n type: anchor.type as any,\n text: anchor.text,\n priority: anchor.priority,\n metadata: anchor.metadata,\n });\n }\n }\n\n private async mergeFrames(\n existingFrame: Frame,\n sourceFrame: Frame,\n targetStack: FrameManager\n ): Promise<void> {\n // Simple merge strategy - append events and anchors\n const sourceEvents = await this.individualStack.getFrameEvents(\n sourceFrame.frame_id\n );\n for (const event of sourceEvents) {\n await targetStack.addEvent(existingFrame.frame_id, {\n type: event.type as any,\n text: event.text,\n metadata: { ...event.metadata, merged: true },\n });\n }\n\n const sourceAnchors = await this.individualStack.getFrameAnchors(\n sourceFrame.frame_id\n );\n for (const anchor of sourceAnchors) {\n await targetStack.addAnchor(existingFrame.frame_id, {\n type: anchor.type as any,\n text: anchor.text,\n priority: anchor.priority,\n metadata: { ...anchor.metadata, merged: true },\n });\n }\n }\n\n private async loadStackContext(\n stackId: string\n ): Promise<StackContext | null> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n return null;\n }\n\n const query = rawDb.prepare(`\n SELECT stack_id, type, project_id, owner_id, team_id, permissions, metadata, created_at, last_active\n FROM stack_contexts \n WHERE stack_id = ?\n `);\n\n const row = query.get(stackId) as any;\n if (!row) {\n return null;\n }\n\n return {\n stackId: row.stack_id,\n type: row.type,\n projectId: row.project_id,\n ownerId: row.owner_id,\n teamId: row.team_id,\n permissions: JSON.parse(row.permissions),\n metadata: JSON.parse(row.metadata || '{}'),\n createdAt: new Date(row.created_at),\n lastActive: new Date(row.last_active),\n };\n } catch (error: unknown) {\n logger.error('Failed to load stack context', { stackId, error });\n return null;\n }\n }\n\n private async saveStackContext(context: StackContext): Promise<void> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n throw new DatabaseError(\n 'SQLite database not available for stack context save',\n ErrorCode.DB_CONNECTION_FAILED,\n { stackId: context.stackId, operation: 'saveStackContext' }\n );\n }\n\n const query = rawDb.prepare(`\n INSERT OR REPLACE INTO stack_contexts \n (stack_id, type, project_id, owner_id, team_id, permissions, metadata, created_at, last_active)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)\n `);\n\n query.run(\n context.stackId,\n context.type,\n context.projectId,\n context.ownerId || null,\n context.teamId || null,\n JSON.stringify(context.permissions),\n JSON.stringify(context.metadata || {}),\n context.createdAt.getTime(),\n context.lastActive.getTime()\n );\n\n logger.debug('Saved stack context', { stackId: context.stackId });\n } catch (error: unknown) {\n logger.error('Failed to save stack context', {\n stackId: context.stackId,\n error,\n });\n throw error;\n }\n }\n\n private async updateStackActivity(stackId: string): Promise<void> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n logger.warn('SQLite database not available for activity update');\n return;\n }\n\n const query = rawDb.prepare(`\n UPDATE stack_contexts \n SET last_active = ?\n WHERE stack_id = ?\n `);\n\n query.run(Date.now(), stackId);\n logger.debug('Updated stack activity', { stackId });\n } catch (error: unknown) {\n logger.error('Failed to update stack activity', { stackId, error });\n // Don't throw - activity updates are not critical\n }\n }\n\n private async loadHandoffRequest(\n requestId: string\n ): Promise<HandoffRequest | null> {\n // Try in-memory first for fast access\n const memoryRequest = this.handoffRequests.get(requestId);\n if (memoryRequest) {\n return memoryRequest;\n }\n\n // Try loading from database\n try {\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n const query = rawDb.prepare(`\n SELECT * FROM handoff_requests WHERE request_id = ?\n `);\n\n const row = query.get(requestId) as any;\n if (row) {\n const request: HandoffRequest = {\n requestId: row.request_id,\n sourceStackId: row.source_stack_id,\n targetStackId: row.target_stack_id,\n frameIds: JSON.parse(row.frame_ids),\n status: row.status,\n createdAt: new Date(row.created_at),\n expiresAt: new Date(row.expires_at),\n targetUserId: row.target_user_id,\n message: row.message,\n };\n\n // Cache in memory for future access\n this.handoffRequests.set(requestId, request);\n return request;\n }\n }\n } catch (error: unknown) {\n logger.error('Failed to load handoff request from database', {\n requestId,\n error,\n });\n }\n\n return null;\n }\n\n private async saveHandoffRequest(request: HandoffRequest): Promise<void> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n const query = rawDb.prepare(`\n INSERT OR REPLACE INTO handoff_requests \n (request_id, source_stack_id, target_stack_id, frame_ids, status, created_at, expires_at, target_user_id, message)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)\n `);\n\n query.run(\n request.requestId,\n request.sourceStackId,\n request.targetStackId,\n JSON.stringify(request.frameIds),\n request.status,\n request.createdAt.getTime(),\n request.expiresAt.getTime(),\n request.targetUserId || null,\n request.message || null\n );\n\n logger.debug('Saved handoff request to database', {\n requestId: request.requestId,\n });\n }\n\n // Also keep in-memory for fast access\n this.handoffRequests.set(request.requestId, request);\n } catch (error: unknown) {\n logger.error('Failed to save handoff request', {\n requestId: request.requestId,\n error,\n });\n // Fallback to in-memory only\n this.handoffRequests.set(request.requestId, request);\n }\n }\n\n /**\n * Get list of available stacks for the current user\n */\n async getAvailableStacks(): Promise<StackContext[]> {\n try {\n const stacks: StackContext[] = [];\n\n // Always include current individual stack context\n stacks.push(this.activeContext);\n\n // Query database for all shared stacks the user has access to\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n const query = rawDb.prepare(`\n SELECT stack_id, type, project_id, owner_id, team_id, permissions, metadata, created_at, last_active\n FROM stack_contexts \n WHERE type = 'shared' AND project_id = ?\n `);\n\n const rows = query.all(this.activeContext.projectId) as any[];\n\n for (const row of rows) {\n const context: StackContext = {\n stackId: row.stack_id,\n type: row.type,\n projectId: row.project_id,\n ownerId: row.owner_id,\n teamId: row.team_id,\n permissions: JSON.parse(row.permissions),\n metadata: JSON.parse(row.metadata || '{}'),\n createdAt: new Date(row.created_at),\n lastActive: new Date(row.last_active),\n };\n\n // Check if user has permission to access this stack\n try {\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n this.activeContext.ownerId || 'unknown',\n 'read',\n 'stack',\n context.stackId,\n context\n )\n );\n stacks.push(context);\n } catch (permissionError: unknown) {\n // Skip stacks user doesn't have access to\n logger.debug('User lacks access to stack', {\n stackId: context.stackId,\n userId: this.activeContext.ownerId,\n });\n }\n }\n }\n\n return stacks;\n } catch (error: unknown) {\n logger.error('Failed to get available stacks', error);\n // Return at least the current individual stack\n return [this.activeContext];\n }\n }\n\n /**\n * Get pending handoff requests for the current user\n */\n async getPendingHandoffRequests(): Promise<HandoffRequest[]> {\n return Array.from(this.handoffRequests.values()).filter(\n (request) =>\n request.status === 'pending' && request.expiresAt > new Date()\n );\n }\n\n /**\n * Get current stack context\n */\n getCurrentContext(): StackContext {\n return { ...this.activeContext };\n }\n\n /**\n * Get permission manager for external access\n */\n getPermissionManager(): PermissionManager {\n return this.permissionManager;\n }\n\n /**\n * Add user to shared stack with specific permissions\n */\n async addUserToSharedStack(\n stackId: string,\n userId: string,\n permissions: StackPermissions,\n requesterId: string\n ): Promise<void> {\n // Check if requester has admin permissions on the stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n requesterId,\n 'administer',\n 'stack',\n stackId\n )\n );\n\n // Grant permissions to the new user\n this.permissionManager.setStackPermissions(userId, stackId, permissions);\n\n logger.info(`Added user to shared stack`, {\n stackId,\n userId,\n permissions,\n requesterId,\n });\n }\n\n /**\n * Remove user from shared stack\n */\n async removeUserFromSharedStack(\n stackId: string,\n userId: string,\n requesterId: string\n ): Promise<void> {\n // Check if requester has admin permissions on the stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n requesterId,\n 'administer',\n 'stack',\n stackId\n )\n );\n\n // Remove user's permissions\n const userPermissions = this.permissionManager.getUserPermissions(userId);\n userPermissions.delete(stackId);\n\n logger.info(`Removed user from shared stack`, {\n stackId,\n userId,\n requesterId,\n });\n }\n}\n"],
5
5
  "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/context/enhanced-rehydration.ts"],
4
- "sourcesContent": ["/**\n * Enhanced Context Rehydration System\n * Addresses compact summary limitations with rich context recovery\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { logger } from '../monitoring/logger.js';\nimport { FrameManager } from './frame-manager.js';\nimport { CompactionHandler } from './compaction-handler.js';\n\nexport interface FileSnapshot {\n path: string;\n content: string;\n size: number;\n lastModified: number;\n hash: string; // Quick change detection\n contextTags: string[]; // e.g., ['migration', 'pipedream', 'hubspot']\n}\n\nexport interface StackTrace {\n error_message: string;\n stack_frames: string[];\n file_path?: string;\n line_number?: number;\n function_name?: string;\n timestamp: number;\n context: string; // What was being done when error occurred\n resolution_attempted?: string[];\n resolution_status: 'pending' | 'resolved' | 'workaround' | 'blocked';\n}\n\nexport interface ConversationContext {\n timestamp: number;\n reasoning: string[];\n decisions_made: string[];\n next_steps: string[];\n user_preferences: Record<string, any>;\n pain_points: string[];\n stack_traces: StackTrace[];\n error_patterns: string[]; // Recurring error types\n}\n\nexport interface ProjectMapping {\n file_relationships: Record<string, string[]>; // file -> related files\n workflow_sequences: string[][]; // sequences of files in workflows\n key_directories: string[];\n entry_points: string[];\n configuration_files: string[];\n}\n\nexport interface RehydrationContext {\n session_id: string;\n compact_detected_at: number;\n pre_compact_state: {\n file_snapshots: FileSnapshot[];\n conversation_context: ConversationContext;\n project_mapping: ProjectMapping;\n active_workflows: string[];\n current_focus: string;\n };\n recovery_anchors: string[];\n}\n\nexport class EnhancedRehydrationManager {\n private frameManager: FrameManager;\n private compactionHandler: CompactionHandler;\n private snapshotThreshold = 10; // Take snapshot every N significant events\n private eventCount = 0;\n private rehydrationStorage = new Map<string, RehydrationContext>();\n\n constructor(frameManager: FrameManager, compactionHandler: CompactionHandler) {\n this.frameManager = frameManager;\n this.compactionHandler = compactionHandler;\n this.setupCompactDetection();\n this.initializeStackTraceStorage();\n }\n\n /**\n * Initialize dedicated stack trace storage in database\n */\n private initializeStackTraceStorage(): void {\n try {\n const db = (this.frameManager as any).db; // Access the underlying database\n \n // Create stack_traces table for persistent storage\n db.exec(`\n CREATE TABLE IF NOT EXISTS stack_traces (\n trace_id TEXT PRIMARY KEY,\n frame_id TEXT,\n project_id TEXT NOT NULL,\n error_message TEXT NOT NULL,\n stack_frames TEXT NOT NULL,\n file_path TEXT,\n line_number INTEGER,\n function_name TEXT,\n context TEXT,\n resolution_attempted TEXT,\n resolution_status TEXT NOT NULL DEFAULT 'pending',\n error_type TEXT,\n error_severity TEXT DEFAULT 'medium',\n created_at INTEGER DEFAULT (unixepoch()),\n updated_at INTEGER DEFAULT (unixepoch()),\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id)\n );\n\n CREATE INDEX IF NOT EXISTS idx_stack_traces_frame ON stack_traces(frame_id);\n CREATE INDEX IF NOT EXISTS idx_stack_traces_status ON stack_traces(resolution_status);\n CREATE INDEX IF NOT EXISTS idx_stack_traces_type ON stack_traces(error_type);\n CREATE INDEX IF NOT EXISTS idx_stack_traces_severity ON stack_traces(error_severity);\n CREATE INDEX IF NOT EXISTS idx_stack_traces_created ON stack_traces(created_at);\n `);\n \n logger.info('Stack trace storage initialized');\n } catch (error) {\n logger.error('Failed to initialize stack trace storage:', error);\n }\n }\n\n /**\n * Set up automatic compact detection and recovery\n */\n private setupCompactDetection(): void {\n // Monitor for compact indicators in new frames\n setInterval(() => this.checkForCompactionEvent(), 30000); // Check every 30s\n }\n\n /**\n * Enhanced file content snapshot with context\n */\n async captureFileSnapshot(filePath: string, contextTags: string[] = []): Promise<FileSnapshot | null> {\n try {\n const stats = await fs.stat(filePath);\n const content = await fs.readFile(filePath, 'utf8');\n \n // Simple hash for change detection\n const hash = this.simpleHash(content);\n\n return {\n path: filePath,\n content: content,\n size: stats.size,\n lastModified: stats.mtimeMs,\n hash: hash,\n contextTags: contextTags\n };\n } catch (error) {\n logger.warn(`Failed to capture snapshot for ${filePath}:`, error);\n return null;\n }\n }\n\n /**\n * Capture conversation reasoning and decisions including stack traces\n */\n captureConversationContext(\n reasoning: string[],\n decisions: string[],\n nextSteps: string[] = [],\n userPrefs: Record<string, any> = {},\n painPoints: string[] = [],\n stackTraces: StackTrace[] = [],\n errorPatterns: string[] = []\n ): ConversationContext {\n return {\n timestamp: Date.now(),\n reasoning: reasoning,\n decisions_made: decisions,\n next_steps: nextSteps,\n user_preferences: userPrefs,\n pain_points: painPoints,\n stack_traces: stackTraces,\n error_patterns: errorPatterns\n };\n }\n\n /**\n * Capture stack trace from error with context and store in database\n */\n captureStackTrace(\n error: Error | string,\n context: string,\n filePath?: string,\n resolutionAttempts: string[] = [],\n frameId?: string\n ): StackTrace {\n const errorMessage = typeof error === 'string' ? error : error.message;\n const stackFrames = typeof error === 'string' ? [] : (error.stack?.split('\\n') || []);\n\n // Extract file path and line number from stack if not provided\n let extractedFilePath = filePath;\n let lineNumber: number | undefined;\n let functionName: string | undefined;\n\n if (stackFrames.length > 0) {\n const firstFrame = stackFrames.find(frame => frame.includes('at '));\n if (firstFrame) {\n const match = firstFrame.match(/at (.+?) \\((.+):(\\d+):(\\d+)\\)/);\n if (match) {\n functionName = match[1];\n extractedFilePath = extractedFilePath || match[2];\n lineNumber = parseInt(match[3]);\n }\n }\n }\n\n const stackTrace: StackTrace = {\n error_message: errorMessage,\n stack_frames: stackFrames,\n file_path: extractedFilePath,\n line_number: lineNumber,\n function_name: functionName,\n timestamp: Date.now(),\n context: context,\n resolution_attempted: resolutionAttempts,\n resolution_status: 'pending'\n };\n\n // Store in database\n this.storeStackTrace(stackTrace, frameId);\n\n return stackTrace;\n }\n\n /**\n * Store stack trace in database\n */\n private storeStackTrace(stackTrace: StackTrace, frameId?: string): string {\n try {\n const db = (this.frameManager as any).db;\n const traceId = this.generateTraceId();\n const currentFrameId = frameId || this.frameManager.getCurrentFrameId();\n \n // Determine error type and severity\n const errorType = this.extractErrorType(stackTrace.error_message);\n const severity = this.determineErrorSeverity(stackTrace);\n\n const stmt = db.prepare(`\n INSERT INTO stack_traces (\n trace_id, frame_id, project_id, error_message, stack_frames,\n file_path, line_number, function_name, context, resolution_attempted,\n resolution_status, error_type, error_severity\n ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n `);\n\n stmt.run(\n traceId,\n currentFrameId,\n (this.frameManager as any).projectId,\n stackTrace.error_message,\n JSON.stringify(stackTrace.stack_frames),\n stackTrace.file_path,\n stackTrace.line_number,\n stackTrace.function_name,\n stackTrace.context,\n JSON.stringify(stackTrace.resolution_attempted),\n stackTrace.resolution_status,\n errorType,\n severity\n );\n\n logger.info(`Stored stack trace ${traceId} for frame ${currentFrameId}`);\n return traceId;\n } catch (error) {\n logger.error('Failed to store stack trace:', error);\n return '';\n }\n }\n\n /**\n * Retrieve stack traces from database\n */\n public getStackTraces(frameId?: string, limit: number = 50): StackTrace[] {\n try {\n const db = (this.frameManager as any).db;\n const traces: StackTrace[] = [];\n\n let query: string;\n let params: any[];\n\n if (frameId) {\n query = `\n SELECT * FROM stack_traces \n WHERE frame_id = ? \n ORDER BY created_at DESC \n LIMIT ?\n `;\n params = [frameId, limit];\n } else {\n query = `\n SELECT * FROM stack_traces \n WHERE project_id = ? \n ORDER BY created_at DESC \n LIMIT ?\n `;\n params = [(this.frameManager as any).projectId, limit];\n }\n\n const rows = db.prepare(query).all(...params);\n\n for (const row of rows) {\n traces.push({\n error_message: row.error_message,\n stack_frames: JSON.parse(row.stack_frames || '[]'),\n file_path: row.file_path,\n line_number: row.line_number,\n function_name: row.function_name,\n timestamp: row.created_at * 1000, // Convert from unix to JS timestamp\n context: row.context,\n resolution_attempted: JSON.parse(row.resolution_attempted || '[]'),\n resolution_status: row.resolution_status\n });\n }\n\n return traces;\n } catch (error) {\n logger.error('Failed to retrieve stack traces:', error);\n return [];\n }\n }\n\n /**\n * Update stack trace resolution status\n */\n public updateStackTraceStatus(\n traceId: string,\n status: StackTrace['resolution_status'],\n resolutionAttempts?: string[]\n ): boolean {\n try {\n const db = (this.frameManager as any).db;\n \n const stmt = db.prepare(`\n UPDATE stack_traces \n SET resolution_status = ?, resolution_attempted = ?, updated_at = unixepoch()\n WHERE trace_id = ?\n `);\n\n const result = stmt.run(\n status,\n resolutionAttempts ? JSON.stringify(resolutionAttempts) : undefined,\n traceId\n );\n\n return result.changes > 0;\n } catch (error) {\n logger.error('Failed to update stack trace status:', error);\n return false;\n }\n }\n\n /**\n * Helper methods for stack trace processing\n */\n private generateTraceId(): string {\n return `trace_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;\n }\n\n private extractErrorType(errorMessage: string): string {\n const typeMatch = errorMessage.match(/^(\\w+Error?):/);\n return typeMatch ? typeMatch[1] : 'Unknown';\n }\n\n private determineErrorSeverity(stackTrace: StackTrace): string {\n const message = stackTrace.error_message.toLowerCase();\n \n if (message.includes('critical') || message.includes('fatal') || message.includes('cannot read properties')) {\n return 'high';\n } else if (message.includes('warning') || message.includes('deprecated')) {\n return 'low';\n } else {\n return 'medium';\n }\n }\n\n /**\n * Auto-detect project structure and relationships\n */\n async analyzeProjectMapping(workingDir: string): Promise<ProjectMapping> {\n const mapping: ProjectMapping = {\n file_relationships: {},\n workflow_sequences: [],\n key_directories: [],\n entry_points: [],\n configuration_files: []\n };\n\n try {\n // Find configuration files\n const configPatterns = [\n 'package.json', 'tsconfig.json', '.env', 'docker-compose.yml',\n '*.config.js', '*.config.ts', 'Dockerfile', 'README.md'\n ];\n\n // Analyze directory structure\n const files = await this.getDirectoryFiles(workingDir);\n \n for (const file of files) {\n const ext = path.extname(file);\n const basename = path.basename(file);\n \n // Identify configuration files\n if (configPatterns.some(pattern => \n pattern.includes('*') ? basename.includes(pattern.replace('*', '')) : basename === pattern\n )) {\n mapping.configuration_files.push(file);\n }\n\n // Identify entry points\n if (basename === 'index.js' || basename === 'index.ts' || basename === 'main.js') {\n mapping.entry_points.push(file);\n }\n\n // Find related files based on naming patterns\n const filePrefix = basename.split('.')[0];\n const relatedFiles = files.filter(f => \n f !== file && path.basename(f).startsWith(filePrefix)\n );\n if (relatedFiles.length > 0) {\n mapping.file_relationships[file] = relatedFiles;\n }\n }\n\n // Identify key directories\n const dirs = files.map(f => path.dirname(f)).filter((v, i, a) => a.indexOf(v) === i);\n mapping.key_directories = dirs.filter(dir => \n ['src', 'lib', 'components', 'pages', 'api', 'utils', 'types'].some(key => dir.includes(key))\n );\n\n } catch (error) {\n logger.warn('Failed to analyze project mapping:', error);\n }\n\n return mapping;\n }\n\n /**\n * Create comprehensive rehydration context before compaction\n */\n async createRehydrationCheckpoint(): Promise<string> {\n const sessionId = this.frameManager.getSessionId() || 'unknown';\n const checkpointId = `${sessionId}_${Date.now()}`;\n\n try {\n // Get current working directory\n const workingDir = process.cwd();\n\n // Capture file snapshots for recently modified files\n const fileSnapshots: FileSnapshot[] = [];\n const recentFiles = await this.getRecentlyModifiedFiles(workingDir);\n \n for (const file of recentFiles.slice(0, 20)) { // Limit to 20 most recent\n const snapshot = await this.captureFileSnapshot(file, this.inferContextTags(file));\n if (snapshot) {\n fileSnapshots.push(snapshot);\n }\n }\n\n // Capture project mapping\n const projectMapping = await this.analyzeProjectMapping(workingDir);\n\n // Extract conversation context from recent events\n const conversationContext = this.extractConversationContext();\n\n // Create rehydration context\n const rehydrationContext: RehydrationContext = {\n session_id: sessionId,\n compact_detected_at: Date.now(),\n pre_compact_state: {\n file_snapshots: fileSnapshots,\n conversation_context: conversationContext,\n project_mapping: projectMapping,\n active_workflows: this.detectActiveWorkflows(fileSnapshots),\n current_focus: this.inferCurrentFocus(fileSnapshots, conversationContext)\n },\n recovery_anchors: this.createRecoveryAnchors(fileSnapshots, conversationContext)\n };\n\n // Store for later retrieval\n this.rehydrationStorage.set(checkpointId, rehydrationContext);\n\n // Also persist to file system for cross-session recovery\n await this.persistRehydrationContext(checkpointId, rehydrationContext);\n\n logger.info(`Created rehydration checkpoint ${checkpointId} with ${fileSnapshots.length} file snapshots`);\n \n return checkpointId;\n } catch (error) {\n logger.error('Failed to create rehydration checkpoint:', error);\n throw error;\n }\n }\n\n /**\n * Inject rich context after compaction detection\n */\n async rehydrateContext(checkpointId?: string): Promise<boolean> {\n try {\n let context: RehydrationContext | undefined;\n\n if (checkpointId) {\n context = this.rehydrationStorage.get(checkpointId);\n if (!context) {\n context = await this.loadPersistedContext(checkpointId);\n }\n } else {\n // Find most recent context\n context = await this.findMostRecentContext();\n }\n\n if (!context) {\n logger.warn('No rehydration context available');\n return false;\n }\n\n await this.injectRichContext(context);\n return true;\n } catch (error) {\n logger.error('Failed to rehydrate context:', error);\n return false;\n }\n }\n\n /**\n * Inject rich context into current session\n */\n private async injectRichContext(context: RehydrationContext): Promise<void> {\n const frameId = this.frameManager.getCurrentFrameId();\n if (!frameId) {\n logger.warn('No active frame for context injection');\n return;\n }\n\n // Inject file context\n for (const snapshot of context.pre_compact_state.file_snapshots.slice(0, 5)) { // Top 5 files\n this.frameManager.addAnchor(\n 'FACT',\n `File: ${snapshot.path} (${snapshot.contextTags.join(', ')})\\n` +\n `Last modified: ${new Date(snapshot.lastModified).toISOString()}\\n` +\n `Size: ${snapshot.size} bytes\\n` +\n `Content preview: ${this.getContentPreview(snapshot.content)}`,\n 9,\n { \n rehydration: true, \n file_path: snapshot.path,\n context_tags: snapshot.contextTags\n },\n frameId\n );\n }\n\n // Inject conversation context\n const conv = context.pre_compact_state.conversation_context;\n if (conv.decisions_made.length > 0) {\n this.frameManager.addAnchor(\n 'DECISION',\n `Previous decisions: ${conv.decisions_made.join('; ')}`,\n 8,\n { rehydration: true },\n frameId\n );\n }\n\n if (conv.next_steps.length > 0) {\n this.frameManager.addAnchor(\n 'FACT',\n `Next steps identified: ${conv.next_steps.join('; ')}`,\n 7,\n { rehydration: true },\n frameId\n );\n }\n\n // Inject stack trace context\n if (conv.stack_traces.length > 0) {\n for (const trace of conv.stack_traces.slice(0, 3)) { // Top 3 most recent errors\n this.frameManager.addAnchor(\n 'ERROR',\n `Error context: ${trace.error_message}\\n` +\n `Context: ${trace.context}\\n` +\n `File: ${trace.file_path || 'unknown'}${trace.line_number ? `:${trace.line_number}` : ''}\\n` +\n `Function: ${trace.function_name || 'unknown'}\\n` +\n `Status: ${trace.resolution_status}\\n` +\n `Stack preview: ${trace.stack_frames.slice(0, 3).join('\\n')}`,\n 9,\n { \n rehydration: true,\n error_type: trace.error_message.split(':')[0],\n resolution_status: trace.resolution_status,\n file_path: trace.file_path\n },\n frameId\n );\n }\n }\n\n // Inject error patterns\n if (conv.error_patterns.length > 0) {\n this.frameManager.addAnchor(\n 'PATTERN',\n `Recurring error patterns detected: ${conv.error_patterns.join(', ')}`,\n 7,\n { rehydration: true },\n frameId\n );\n }\n\n // Inject project mapping\n const mapping = context.pre_compact_state.project_mapping;\n if (mapping.entry_points.length > 0) {\n this.frameManager.addAnchor(\n 'FACT',\n `Project entry points: ${mapping.entry_points.join(', ')}`,\n 6,\n { rehydration: true },\n frameId\n );\n }\n\n // Inject current focus\n if (context.pre_compact_state.current_focus) {\n this.frameManager.addAnchor(\n 'CONSTRAINT',\n `Previous focus: ${context.pre_compact_state.current_focus}`,\n 8,\n { rehydration: true },\n frameId\n );\n }\n\n logger.info('Rich context injected successfully');\n }\n\n // Helper methods\n private async getDirectoryFiles(dir: string): Promise<string[]> {\n // Implementation to recursively get files\n return []; // Simplified for now\n }\n\n private async getRecentlyModifiedFiles(dir: string): Promise<string[]> {\n // Implementation to get recently modified files\n return []; // Simplified for now\n }\n\n private inferContextTags(filePath: string): string[] {\n const tags: string[] = [];\n const content = filePath.toLowerCase();\n \n if (content.includes('pipeline') || content.includes('migrate')) tags.push('migration');\n if (content.includes('hubspot')) tags.push('hubspot');\n if (content.includes('pipedream')) tags.push('pipedream');\n if (content.includes('test')) tags.push('test');\n if (content.includes('config')) tags.push('configuration');\n \n return tags;\n }\n\n private extractConversationContext(): ConversationContext {\n // Extract from recent frame events\n const recentErrors = this.extractRecentStackTraces();\n const errorPatterns = this.detectErrorPatterns(recentErrors);\n \n return {\n timestamp: Date.now(),\n reasoning: [],\n decisions_made: [],\n next_steps: [],\n user_preferences: {},\n pain_points: [],\n stack_traces: recentErrors,\n error_patterns: errorPatterns\n };\n }\n\n /**\n * Extract recent stack traces from database and frame events\n */\n private extractRecentStackTraces(): StackTrace[] {\n try {\n // Get recent stack traces from database (most reliable source)\n const dbTraces = this.getStackTraces(undefined, 10);\n \n // Also check frame events for additional traces\n const eventTraces = this.extractStackTracesFromFrameEvents();\n \n // Combine and deduplicate\n const allTraces = [...dbTraces, ...eventTraces];\n \n // Remove duplicates based on error message and file path\n const uniqueTraces = allTraces.filter((trace, index, array) => \n array.findIndex(t => \n t.error_message === trace.error_message && \n t.file_path === trace.file_path\n ) === index\n );\n \n // Sort by timestamp (newest first) and return top 5\n return uniqueTraces\n .sort((a, b) => b.timestamp - a.timestamp)\n .slice(0, 5);\n } catch (error) {\n logger.warn('Failed to extract stack traces:', error);\n return [];\n }\n }\n\n /**\n * Extract stack traces from frame events (fallback method)\n */\n private extractStackTracesFromFrameEvents(): StackTrace[] {\n const traces: StackTrace[] = [];\n \n try {\n // Get recent frames and look for error events\n const frames = this.frameManager.getActiveFramePath();\n \n for (const frame of frames.slice(-3)) { // Check last 3 frames\n const frameData = this.frameManager.getFrame(frame.frame_id);\n if (frameData?.events) {\n for (const event of frameData.events) {\n if (event.type === 'error' || event.type === 'exception') {\n const trace = this.parseStackTraceFromEvent(event);\n if (trace) {\n traces.push(trace);\n }\n }\n }\n }\n }\n } catch (error) {\n logger.warn('Failed to extract frame event traces:', error);\n }\n \n return traces;\n }\n\n /**\n * Parse stack trace from frame event\n */\n private parseStackTraceFromEvent(event: any): StackTrace | null {\n try {\n const data = typeof event.data === 'string' ? JSON.parse(event.data) : event.data;\n \n return {\n error_message: data.error || data.message || 'Unknown error',\n stack_frames: data.stack ? data.stack.split('\\n') : [],\n file_path: data.file || data.fileName,\n line_number: data.line || data.lineNumber,\n function_name: data.function || data.functionName,\n timestamp: event.timestamp || Date.now(),\n context: data.context || 'Error occurred during frame processing',\n resolution_attempted: data.resolutionAttempts || [],\n resolution_status: data.resolved ? 'resolved' : 'pending'\n };\n } catch (error) {\n return null;\n }\n }\n\n /**\n * Detect recurring error patterns\n */\n private detectErrorPatterns(traces: StackTrace[]): string[] {\n const patterns = new Map<string, number>();\n \n for (const trace of traces) {\n // Extract error type from message\n const errorType = trace.error_message.split(':')[0].trim();\n patterns.set(errorType, (patterns.get(errorType) || 0) + 1);\n }\n \n // Return patterns that occur more than once\n return Array.from(patterns.entries())\n .filter(([, count]) => count > 1)\n .map(([pattern]) => pattern);\n }\n\n private detectActiveWorkflows(snapshots: FileSnapshot[]): string[] {\n const workflows: string[] = [];\n \n for (const snapshot of snapshots) {\n if (snapshot.contextTags.includes('migration')) {\n workflows.push('data_migration');\n }\n if (snapshot.path.includes('test')) {\n workflows.push('testing');\n }\n }\n \n return [...new Set(workflows)];\n }\n\n private inferCurrentFocus(snapshots: FileSnapshot[], context: ConversationContext): string {\n // Analyze recent file activity and conversation to infer focus\n if (snapshots.some(s => s.contextTags.includes('migration'))) {\n return 'Data migration and transformation';\n }\n if (snapshots.some(s => s.path.includes('test'))) {\n return 'Testing and validation';\n }\n return 'Development';\n }\n\n private createRecoveryAnchors(snapshots: FileSnapshot[], context: ConversationContext): string[] {\n const anchors: string[] = [];\n \n // Create anchor points for each significant file\n for (const snapshot of snapshots.slice(0, 3)) {\n anchors.push(`File context: ${snapshot.path} with ${snapshot.contextTags.join(', ')}`);\n }\n \n return anchors;\n }\n\n private async persistRehydrationContext(id: string, context: RehydrationContext): Promise<void> {\n // Implementation to persist context to filesystem\n const contextDir = path.join(process.cwd(), '.stackmemory', 'rehydration');\n await fs.mkdir(contextDir, { recursive: true });\n await fs.writeFile(\n path.join(contextDir, `${id}.json`),\n JSON.stringify(context, null, 2)\n );\n }\n\n private async loadPersistedContext(id: string): Promise<RehydrationContext | undefined> {\n try {\n const contextPath = path.join(process.cwd(), '.stackmemory', 'rehydration', `${id}.json`);\n const content = await fs.readFile(contextPath, 'utf8');\n return JSON.parse(content);\n } catch {\n return undefined;\n }\n }\n\n private async findMostRecentContext(): Promise<RehydrationContext | undefined> {\n // Find most recent persisted context\n return undefined; // Simplified for now\n }\n\n private checkForCompactionEvent(): void {\n // Check if compaction occurred and trigger rehydration\n if (this.compactionHandler.detectCompactionEvent('')) {\n this.rehydrateContext();\n }\n }\n\n private simpleHash(content: string): string {\n let hash = 0;\n for (let i = 0; i < content.length; i++) {\n const char = content.charCodeAt(i);\n hash = ((hash << 5) - hash) + char;\n hash = hash & hash; // Convert to 32bit integer\n }\n return hash.toString(16);\n }\n\n private getContentPreview(content: string, maxLength = 200): string {\n return content.length > maxLength \n ? content.substring(0, maxLength) + '...'\n : content;\n }\n}"],
4
+ "sourcesContent": ["/**\n * Enhanced Context Rehydration System\n * Addresses compact summary limitations with rich context recovery\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { logger } from '../monitoring/logger.js';\nimport { FrameManager } from './index.js';\nimport { CompactionHandler } from './compaction-handler.js';\n\nexport interface FileSnapshot {\n path: string;\n content: string;\n size: number;\n lastModified: number;\n hash: string; // Quick change detection\n contextTags: string[]; // e.g., ['migration', 'pipedream', 'hubspot']\n}\n\nexport interface StackTrace {\n error_message: string;\n stack_frames: string[];\n file_path?: string;\n line_number?: number;\n function_name?: string;\n timestamp: number;\n context: string; // What was being done when error occurred\n resolution_attempted?: string[];\n resolution_status: 'pending' | 'resolved' | 'workaround' | 'blocked';\n}\n\nexport interface ConversationContext {\n timestamp: number;\n reasoning: string[];\n decisions_made: string[];\n next_steps: string[];\n user_preferences: Record<string, any>;\n pain_points: string[];\n stack_traces: StackTrace[];\n error_patterns: string[]; // Recurring error types\n}\n\nexport interface ProjectMapping {\n file_relationships: Record<string, string[]>; // file -> related files\n workflow_sequences: string[][]; // sequences of files in workflows\n key_directories: string[];\n entry_points: string[];\n configuration_files: string[];\n}\n\nexport interface RehydrationContext {\n session_id: string;\n compact_detected_at: number;\n pre_compact_state: {\n file_snapshots: FileSnapshot[];\n conversation_context: ConversationContext;\n project_mapping: ProjectMapping;\n active_workflows: string[];\n current_focus: string;\n };\n recovery_anchors: string[];\n}\n\nexport class EnhancedRehydrationManager {\n private frameManager: FrameManager;\n private compactionHandler: CompactionHandler;\n private snapshotThreshold = 10; // Take snapshot every N significant events\n private eventCount = 0;\n private rehydrationStorage = new Map<string, RehydrationContext>();\n\n constructor(frameManager: FrameManager, compactionHandler: CompactionHandler) {\n this.frameManager = frameManager;\n this.compactionHandler = compactionHandler;\n this.setupCompactDetection();\n this.initializeStackTraceStorage();\n }\n\n /**\n * Initialize dedicated stack trace storage in database\n */\n private initializeStackTraceStorage(): void {\n try {\n const db = (this.frameManager as any).db; // Access the underlying database\n \n // Create stack_traces table for persistent storage\n db.exec(`\n CREATE TABLE IF NOT EXISTS stack_traces (\n trace_id TEXT PRIMARY KEY,\n frame_id TEXT,\n project_id TEXT NOT NULL,\n error_message TEXT NOT NULL,\n stack_frames TEXT NOT NULL,\n file_path TEXT,\n line_number INTEGER,\n function_name TEXT,\n context TEXT,\n resolution_attempted TEXT,\n resolution_status TEXT NOT NULL DEFAULT 'pending',\n error_type TEXT,\n error_severity TEXT DEFAULT 'medium',\n created_at INTEGER DEFAULT (unixepoch()),\n updated_at INTEGER DEFAULT (unixepoch()),\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id)\n );\n\n CREATE INDEX IF NOT EXISTS idx_stack_traces_frame ON stack_traces(frame_id);\n CREATE INDEX IF NOT EXISTS idx_stack_traces_status ON stack_traces(resolution_status);\n CREATE INDEX IF NOT EXISTS idx_stack_traces_type ON stack_traces(error_type);\n CREATE INDEX IF NOT EXISTS idx_stack_traces_severity ON stack_traces(error_severity);\n CREATE INDEX IF NOT EXISTS idx_stack_traces_created ON stack_traces(created_at);\n `);\n \n logger.info('Stack trace storage initialized');\n } catch (error) {\n logger.error('Failed to initialize stack trace storage:', error);\n }\n }\n\n /**\n * Set up automatic compact detection and recovery\n */\n private setupCompactDetection(): void {\n // Monitor for compact indicators in new frames\n setInterval(() => this.checkForCompactionEvent(), 30000); // Check every 30s\n }\n\n /**\n * Enhanced file content snapshot with context\n */\n async captureFileSnapshot(filePath: string, contextTags: string[] = []): Promise<FileSnapshot | null> {\n try {\n const stats = await fs.stat(filePath);\n const content = await fs.readFile(filePath, 'utf8');\n \n // Simple hash for change detection\n const hash = this.simpleHash(content);\n\n return {\n path: filePath,\n content: content,\n size: stats.size,\n lastModified: stats.mtimeMs,\n hash: hash,\n contextTags: contextTags\n };\n } catch (error) {\n logger.warn(`Failed to capture snapshot for ${filePath}:`, error);\n return null;\n }\n }\n\n /**\n * Capture conversation reasoning and decisions including stack traces\n */\n captureConversationContext(\n reasoning: string[],\n decisions: string[],\n nextSteps: string[] = [],\n userPrefs: Record<string, any> = {},\n painPoints: string[] = [],\n stackTraces: StackTrace[] = [],\n errorPatterns: string[] = []\n ): ConversationContext {\n return {\n timestamp: Date.now(),\n reasoning: reasoning,\n decisions_made: decisions,\n next_steps: nextSteps,\n user_preferences: userPrefs,\n pain_points: painPoints,\n stack_traces: stackTraces,\n error_patterns: errorPatterns\n };\n }\n\n /**\n * Capture stack trace from error with context and store in database\n */\n captureStackTrace(\n error: Error | string,\n context: string,\n filePath?: string,\n resolutionAttempts: string[] = [],\n frameId?: string\n ): StackTrace {\n const errorMessage = typeof error === 'string' ? error : error.message;\n const stackFrames = typeof error === 'string' ? [] : (error.stack?.split('\\n') || []);\n\n // Extract file path and line number from stack if not provided\n let extractedFilePath = filePath;\n let lineNumber: number | undefined;\n let functionName: string | undefined;\n\n if (stackFrames.length > 0) {\n const firstFrame = stackFrames.find(frame => frame.includes('at '));\n if (firstFrame) {\n const match = firstFrame.match(/at (.+?) \\((.+):(\\d+):(\\d+)\\)/);\n if (match) {\n functionName = match[1];\n extractedFilePath = extractedFilePath || match[2];\n lineNumber = parseInt(match[3]);\n }\n }\n }\n\n const stackTrace: StackTrace = {\n error_message: errorMessage,\n stack_frames: stackFrames,\n file_path: extractedFilePath,\n line_number: lineNumber,\n function_name: functionName,\n timestamp: Date.now(),\n context: context,\n resolution_attempted: resolutionAttempts,\n resolution_status: 'pending'\n };\n\n // Store in database\n this.storeStackTrace(stackTrace, frameId);\n\n return stackTrace;\n }\n\n /**\n * Store stack trace in database\n */\n private storeStackTrace(stackTrace: StackTrace, frameId?: string): string {\n try {\n const db = (this.frameManager as any).db;\n const traceId = this.generateTraceId();\n const currentFrameId = frameId || this.frameManager.getCurrentFrameId();\n \n // Determine error type and severity\n const errorType = this.extractErrorType(stackTrace.error_message);\n const severity = this.determineErrorSeverity(stackTrace);\n\n const stmt = db.prepare(`\n INSERT INTO stack_traces (\n trace_id, frame_id, project_id, error_message, stack_frames,\n file_path, line_number, function_name, context, resolution_attempted,\n resolution_status, error_type, error_severity\n ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n `);\n\n stmt.run(\n traceId,\n currentFrameId,\n (this.frameManager as any).projectId,\n stackTrace.error_message,\n JSON.stringify(stackTrace.stack_frames),\n stackTrace.file_path,\n stackTrace.line_number,\n stackTrace.function_name,\n stackTrace.context,\n JSON.stringify(stackTrace.resolution_attempted),\n stackTrace.resolution_status,\n errorType,\n severity\n );\n\n logger.info(`Stored stack trace ${traceId} for frame ${currentFrameId}`);\n return traceId;\n } catch (error) {\n logger.error('Failed to store stack trace:', error);\n return '';\n }\n }\n\n /**\n * Retrieve stack traces from database\n */\n public getStackTraces(frameId?: string, limit: number = 50): StackTrace[] {\n try {\n const db = (this.frameManager as any).db;\n const traces: StackTrace[] = [];\n\n let query: string;\n let params: any[];\n\n if (frameId) {\n query = `\n SELECT * FROM stack_traces \n WHERE frame_id = ? \n ORDER BY created_at DESC \n LIMIT ?\n `;\n params = [frameId, limit];\n } else {\n query = `\n SELECT * FROM stack_traces \n WHERE project_id = ? \n ORDER BY created_at DESC \n LIMIT ?\n `;\n params = [(this.frameManager as any).projectId, limit];\n }\n\n const rows = db.prepare(query).all(...params);\n\n for (const row of rows) {\n traces.push({\n error_message: row.error_message,\n stack_frames: JSON.parse(row.stack_frames || '[]'),\n file_path: row.file_path,\n line_number: row.line_number,\n function_name: row.function_name,\n timestamp: row.created_at * 1000, // Convert from unix to JS timestamp\n context: row.context,\n resolution_attempted: JSON.parse(row.resolution_attempted || '[]'),\n resolution_status: row.resolution_status\n });\n }\n\n return traces;\n } catch (error) {\n logger.error('Failed to retrieve stack traces:', error);\n return [];\n }\n }\n\n /**\n * Update stack trace resolution status\n */\n public updateStackTraceStatus(\n traceId: string,\n status: StackTrace['resolution_status'],\n resolutionAttempts?: string[]\n ): boolean {\n try {\n const db = (this.frameManager as any).db;\n \n const stmt = db.prepare(`\n UPDATE stack_traces \n SET resolution_status = ?, resolution_attempted = ?, updated_at = unixepoch()\n WHERE trace_id = ?\n `);\n\n const result = stmt.run(\n status,\n resolutionAttempts ? JSON.stringify(resolutionAttempts) : undefined,\n traceId\n );\n\n return result.changes > 0;\n } catch (error) {\n logger.error('Failed to update stack trace status:', error);\n return false;\n }\n }\n\n /**\n * Helper methods for stack trace processing\n */\n private generateTraceId(): string {\n return `trace_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;\n }\n\n private extractErrorType(errorMessage: string): string {\n const typeMatch = errorMessage.match(/^(\\w+Error?):/);\n return typeMatch ? typeMatch[1] : 'Unknown';\n }\n\n private determineErrorSeverity(stackTrace: StackTrace): string {\n const message = stackTrace.error_message.toLowerCase();\n \n if (message.includes('critical') || message.includes('fatal') || message.includes('cannot read properties')) {\n return 'high';\n } else if (message.includes('warning') || message.includes('deprecated')) {\n return 'low';\n } else {\n return 'medium';\n }\n }\n\n /**\n * Auto-detect project structure and relationships\n */\n async analyzeProjectMapping(workingDir: string): Promise<ProjectMapping> {\n const mapping: ProjectMapping = {\n file_relationships: {},\n workflow_sequences: [],\n key_directories: [],\n entry_points: [],\n configuration_files: []\n };\n\n try {\n // Find configuration files\n const configPatterns = [\n 'package.json', 'tsconfig.json', '.env', 'docker-compose.yml',\n '*.config.js', '*.config.ts', 'Dockerfile', 'README.md'\n ];\n\n // Analyze directory structure\n const files = await this.getDirectoryFiles(workingDir);\n \n for (const file of files) {\n const ext = path.extname(file);\n const basename = path.basename(file);\n \n // Identify configuration files\n if (configPatterns.some(pattern => \n pattern.includes('*') ? basename.includes(pattern.replace('*', '')) : basename === pattern\n )) {\n mapping.configuration_files.push(file);\n }\n\n // Identify entry points\n if (basename === 'index.js' || basename === 'index.ts' || basename === 'main.js') {\n mapping.entry_points.push(file);\n }\n\n // Find related files based on naming patterns\n const filePrefix = basename.split('.')[0];\n const relatedFiles = files.filter(f => \n f !== file && path.basename(f).startsWith(filePrefix)\n );\n if (relatedFiles.length > 0) {\n mapping.file_relationships[file] = relatedFiles;\n }\n }\n\n // Identify key directories\n const dirs = files.map(f => path.dirname(f)).filter((v, i, a) => a.indexOf(v) === i);\n mapping.key_directories = dirs.filter(dir => \n ['src', 'lib', 'components', 'pages', 'api', 'utils', 'types'].some(key => dir.includes(key))\n );\n\n } catch (error) {\n logger.warn('Failed to analyze project mapping:', error);\n }\n\n return mapping;\n }\n\n /**\n * Create comprehensive rehydration context before compaction\n */\n async createRehydrationCheckpoint(): Promise<string> {\n const sessionId = this.frameManager.getSessionId() || 'unknown';\n const checkpointId = `${sessionId}_${Date.now()}`;\n\n try {\n // Get current working directory\n const workingDir = process.cwd();\n\n // Capture file snapshots for recently modified files\n const fileSnapshots: FileSnapshot[] = [];\n const recentFiles = await this.getRecentlyModifiedFiles(workingDir);\n \n for (const file of recentFiles.slice(0, 20)) { // Limit to 20 most recent\n const snapshot = await this.captureFileSnapshot(file, this.inferContextTags(file));\n if (snapshot) {\n fileSnapshots.push(snapshot);\n }\n }\n\n // Capture project mapping\n const projectMapping = await this.analyzeProjectMapping(workingDir);\n\n // Extract conversation context from recent events\n const conversationContext = this.extractConversationContext();\n\n // Create rehydration context\n const rehydrationContext: RehydrationContext = {\n session_id: sessionId,\n compact_detected_at: Date.now(),\n pre_compact_state: {\n file_snapshots: fileSnapshots,\n conversation_context: conversationContext,\n project_mapping: projectMapping,\n active_workflows: this.detectActiveWorkflows(fileSnapshots),\n current_focus: this.inferCurrentFocus(fileSnapshots, conversationContext)\n },\n recovery_anchors: this.createRecoveryAnchors(fileSnapshots, conversationContext)\n };\n\n // Store for later retrieval\n this.rehydrationStorage.set(checkpointId, rehydrationContext);\n\n // Also persist to file system for cross-session recovery\n await this.persistRehydrationContext(checkpointId, rehydrationContext);\n\n logger.info(`Created rehydration checkpoint ${checkpointId} with ${fileSnapshots.length} file snapshots`);\n \n return checkpointId;\n } catch (error) {\n logger.error('Failed to create rehydration checkpoint:', error);\n throw error;\n }\n }\n\n /**\n * Inject rich context after compaction detection\n */\n async rehydrateContext(checkpointId?: string): Promise<boolean> {\n try {\n let context: RehydrationContext | undefined;\n\n if (checkpointId) {\n context = this.rehydrationStorage.get(checkpointId);\n if (!context) {\n context = await this.loadPersistedContext(checkpointId);\n }\n } else {\n // Find most recent context\n context = await this.findMostRecentContext();\n }\n\n if (!context) {\n logger.warn('No rehydration context available');\n return false;\n }\n\n await this.injectRichContext(context);\n return true;\n } catch (error) {\n logger.error('Failed to rehydrate context:', error);\n return false;\n }\n }\n\n /**\n * Inject rich context into current session\n */\n private async injectRichContext(context: RehydrationContext): Promise<void> {\n const frameId = this.frameManager.getCurrentFrameId();\n if (!frameId) {\n logger.warn('No active frame for context injection');\n return;\n }\n\n // Inject file context\n for (const snapshot of context.pre_compact_state.file_snapshots.slice(0, 5)) { // Top 5 files\n this.frameManager.addAnchor(\n 'FACT',\n `File: ${snapshot.path} (${snapshot.contextTags.join(', ')})\\n` +\n `Last modified: ${new Date(snapshot.lastModified).toISOString()}\\n` +\n `Size: ${snapshot.size} bytes\\n` +\n `Content preview: ${this.getContentPreview(snapshot.content)}`,\n 9,\n { \n rehydration: true, \n file_path: snapshot.path,\n context_tags: snapshot.contextTags\n },\n frameId\n );\n }\n\n // Inject conversation context\n const conv = context.pre_compact_state.conversation_context;\n if (conv.decisions_made.length > 0) {\n this.frameManager.addAnchor(\n 'DECISION',\n `Previous decisions: ${conv.decisions_made.join('; ')}`,\n 8,\n { rehydration: true },\n frameId\n );\n }\n\n if (conv.next_steps.length > 0) {\n this.frameManager.addAnchor(\n 'FACT',\n `Next steps identified: ${conv.next_steps.join('; ')}`,\n 7,\n { rehydration: true },\n frameId\n );\n }\n\n // Inject stack trace context\n if (conv.stack_traces.length > 0) {\n for (const trace of conv.stack_traces.slice(0, 3)) { // Top 3 most recent errors\n this.frameManager.addAnchor(\n 'ERROR',\n `Error context: ${trace.error_message}\\n` +\n `Context: ${trace.context}\\n` +\n `File: ${trace.file_path || 'unknown'}${trace.line_number ? `:${trace.line_number}` : ''}\\n` +\n `Function: ${trace.function_name || 'unknown'}\\n` +\n `Status: ${trace.resolution_status}\\n` +\n `Stack preview: ${trace.stack_frames.slice(0, 3).join('\\n')}`,\n 9,\n { \n rehydration: true,\n error_type: trace.error_message.split(':')[0],\n resolution_status: trace.resolution_status,\n file_path: trace.file_path\n },\n frameId\n );\n }\n }\n\n // Inject error patterns\n if (conv.error_patterns.length > 0) {\n this.frameManager.addAnchor(\n 'PATTERN',\n `Recurring error patterns detected: ${conv.error_patterns.join(', ')}`,\n 7,\n { rehydration: true },\n frameId\n );\n }\n\n // Inject project mapping\n const mapping = context.pre_compact_state.project_mapping;\n if (mapping.entry_points.length > 0) {\n this.frameManager.addAnchor(\n 'FACT',\n `Project entry points: ${mapping.entry_points.join(', ')}`,\n 6,\n { rehydration: true },\n frameId\n );\n }\n\n // Inject current focus\n if (context.pre_compact_state.current_focus) {\n this.frameManager.addAnchor(\n 'CONSTRAINT',\n `Previous focus: ${context.pre_compact_state.current_focus}`,\n 8,\n { rehydration: true },\n frameId\n );\n }\n\n logger.info('Rich context injected successfully');\n }\n\n // Helper methods\n private async getDirectoryFiles(dir: string): Promise<string[]> {\n // Implementation to recursively get files\n return []; // Simplified for now\n }\n\n private async getRecentlyModifiedFiles(dir: string): Promise<string[]> {\n // Implementation to get recently modified files\n return []; // Simplified for now\n }\n\n private inferContextTags(filePath: string): string[] {\n const tags: string[] = [];\n const content = filePath.toLowerCase();\n \n if (content.includes('pipeline') || content.includes('migrate')) tags.push('migration');\n if (content.includes('hubspot')) tags.push('hubspot');\n if (content.includes('pipedream')) tags.push('pipedream');\n if (content.includes('test')) tags.push('test');\n if (content.includes('config')) tags.push('configuration');\n \n return tags;\n }\n\n private extractConversationContext(): ConversationContext {\n // Extract from recent frame events\n const recentErrors = this.extractRecentStackTraces();\n const errorPatterns = this.detectErrorPatterns(recentErrors);\n \n return {\n timestamp: Date.now(),\n reasoning: [],\n decisions_made: [],\n next_steps: [],\n user_preferences: {},\n pain_points: [],\n stack_traces: recentErrors,\n error_patterns: errorPatterns\n };\n }\n\n /**\n * Extract recent stack traces from database and frame events\n */\n private extractRecentStackTraces(): StackTrace[] {\n try {\n // Get recent stack traces from database (most reliable source)\n const dbTraces = this.getStackTraces(undefined, 10);\n \n // Also check frame events for additional traces\n const eventTraces = this.extractStackTracesFromFrameEvents();\n \n // Combine and deduplicate\n const allTraces = [...dbTraces, ...eventTraces];\n \n // Remove duplicates based on error message and file path\n const uniqueTraces = allTraces.filter((trace, index, array) => \n array.findIndex(t => \n t.error_message === trace.error_message && \n t.file_path === trace.file_path\n ) === index\n );\n \n // Sort by timestamp (newest first) and return top 5\n return uniqueTraces\n .sort((a, b) => b.timestamp - a.timestamp)\n .slice(0, 5);\n } catch (error) {\n logger.warn('Failed to extract stack traces:', error);\n return [];\n }\n }\n\n /**\n * Extract stack traces from frame events (fallback method)\n */\n private extractStackTracesFromFrameEvents(): StackTrace[] {\n const traces: StackTrace[] = [];\n \n try {\n // Get recent frames and look for error events\n const frames = this.frameManager.getActiveFramePath();\n \n for (const frame of frames.slice(-3)) { // Check last 3 frames\n const frameData = this.frameManager.getFrame(frame.frame_id);\n if (frameData?.events) {\n for (const event of frameData.events) {\n if (event.type === 'error' || event.type === 'exception') {\n const trace = this.parseStackTraceFromEvent(event);\n if (trace) {\n traces.push(trace);\n }\n }\n }\n }\n }\n } catch (error) {\n logger.warn('Failed to extract frame event traces:', error);\n }\n \n return traces;\n }\n\n /**\n * Parse stack trace from frame event\n */\n private parseStackTraceFromEvent(event: any): StackTrace | null {\n try {\n const data = typeof event.data === 'string' ? JSON.parse(event.data) : event.data;\n \n return {\n error_message: data.error || data.message || 'Unknown error',\n stack_frames: data.stack ? data.stack.split('\\n') : [],\n file_path: data.file || data.fileName,\n line_number: data.line || data.lineNumber,\n function_name: data.function || data.functionName,\n timestamp: event.timestamp || Date.now(),\n context: data.context || 'Error occurred during frame processing',\n resolution_attempted: data.resolutionAttempts || [],\n resolution_status: data.resolved ? 'resolved' : 'pending'\n };\n } catch (error) {\n return null;\n }\n }\n\n /**\n * Detect recurring error patterns\n */\n private detectErrorPatterns(traces: StackTrace[]): string[] {\n const patterns = new Map<string, number>();\n \n for (const trace of traces) {\n // Extract error type from message\n const errorType = trace.error_message.split(':')[0].trim();\n patterns.set(errorType, (patterns.get(errorType) || 0) + 1);\n }\n \n // Return patterns that occur more than once\n return Array.from(patterns.entries())\n .filter(([, count]) => count > 1)\n .map(([pattern]) => pattern);\n }\n\n private detectActiveWorkflows(snapshots: FileSnapshot[]): string[] {\n const workflows: string[] = [];\n \n for (const snapshot of snapshots) {\n if (snapshot.contextTags.includes('migration')) {\n workflows.push('data_migration');\n }\n if (snapshot.path.includes('test')) {\n workflows.push('testing');\n }\n }\n \n return [...new Set(workflows)];\n }\n\n private inferCurrentFocus(snapshots: FileSnapshot[], context: ConversationContext): string {\n // Analyze recent file activity and conversation to infer focus\n if (snapshots.some(s => s.contextTags.includes('migration'))) {\n return 'Data migration and transformation';\n }\n if (snapshots.some(s => s.path.includes('test'))) {\n return 'Testing and validation';\n }\n return 'Development';\n }\n\n private createRecoveryAnchors(snapshots: FileSnapshot[], context: ConversationContext): string[] {\n const anchors: string[] = [];\n \n // Create anchor points for each significant file\n for (const snapshot of snapshots.slice(0, 3)) {\n anchors.push(`File context: ${snapshot.path} with ${snapshot.contextTags.join(', ')}`);\n }\n \n return anchors;\n }\n\n private async persistRehydrationContext(id: string, context: RehydrationContext): Promise<void> {\n // Implementation to persist context to filesystem\n const contextDir = path.join(process.cwd(), '.stackmemory', 'rehydration');\n await fs.mkdir(contextDir, { recursive: true });\n await fs.writeFile(\n path.join(contextDir, `${id}.json`),\n JSON.stringify(context, null, 2)\n );\n }\n\n private async loadPersistedContext(id: string): Promise<RehydrationContext | undefined> {\n try {\n const contextPath = path.join(process.cwd(), '.stackmemory', 'rehydration', `${id}.json`);\n const content = await fs.readFile(contextPath, 'utf8');\n return JSON.parse(content);\n } catch {\n return undefined;\n }\n }\n\n private async findMostRecentContext(): Promise<RehydrationContext | undefined> {\n // Find most recent persisted context\n return undefined; // Simplified for now\n }\n\n private checkForCompactionEvent(): void {\n // Check if compaction occurred and trigger rehydration\n if (this.compactionHandler.detectCompactionEvent('')) {\n this.rehydrateContext();\n }\n }\n\n private simpleHash(content: string): string {\n let hash = 0;\n for (let i = 0; i < content.length; i++) {\n const char = content.charCodeAt(i);\n hash = ((hash << 5) - hash) + char;\n hash = hash & hash; // Convert to 32bit integer\n }\n return hash.toString(16);\n }\n\n private getContentPreview(content: string, maxLength = 200): string {\n return content.length > maxLength \n ? content.substring(0, maxLength) + '...'\n : content;\n }\n}"],
5
5
  "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -4,6 +4,17 @@ const __filename = __fileURLToPath(import.meta.url);
4
4
  const __dirname = __pathDirname(__filename);
5
5
  import { logger } from "../monitoring/logger.js";
6
6
  import { DatabaseError, ErrorCode } from "../errors/index.js";
7
+ function safeJsonParse(json, fallback) {
8
+ if (!json) return fallback;
9
+ try {
10
+ return JSON.parse(json);
11
+ } catch {
12
+ logger.warn("Failed to parse JSON, using fallback", {
13
+ json: json.substring(0, 100)
14
+ });
15
+ return fallback;
16
+ }
17
+ }
7
18
  class FrameDatabase {
8
19
  constructor(db) {
9
20
  this.db = db;
@@ -135,9 +146,13 @@ class FrameDatabase {
135
146
  if (!row) return void 0;
136
147
  return {
137
148
  ...row,
138
- inputs: JSON.parse(row.inputs || "{}"),
139
- outputs: JSON.parse(row.outputs || "{}"),
140
- digest_json: JSON.parse(row.digest_json || "{}")
149
+ parent_frame_id: row.parent_frame_id ?? void 0,
150
+ inputs: safeJsonParse(row.inputs, {}),
151
+ outputs: safeJsonParse(row.outputs, {}),
152
+ digest_json: safeJsonParse(
153
+ row.digest_json,
154
+ {}
155
+ )
141
156
  };
142
157
  } catch (error) {
143
158
  logger.warn(`Failed to get frame: ${frameId}`, { error });
@@ -171,6 +186,14 @@ class FrameDatabase {
171
186
  setClauses.push("closed_at = ?");
172
187
  values.push(updates.closed_at);
173
188
  }
189
+ if (updates.parent_frame_id !== void 0) {
190
+ setClauses.push("parent_frame_id = ?");
191
+ values.push(updates.parent_frame_id);
192
+ }
193
+ if (updates.depth !== void 0) {
194
+ setClauses.push("depth = ?");
195
+ values.push(updates.depth);
196
+ }
174
197
  if (setClauses.length === 0) {
175
198
  return;
176
199
  }
@@ -205,9 +228,13 @@ class FrameDatabase {
205
228
  const rows = this.db.prepare(query).all(...params);
206
229
  return rows.map((row) => ({
207
230
  ...row,
208
- inputs: JSON.parse(row.inputs || "{}"),
209
- outputs: JSON.parse(row.outputs || "{}"),
210
- digest_json: JSON.parse(row.digest_json || "{}")
231
+ parent_frame_id: row.parent_frame_id ?? void 0,
232
+ inputs: safeJsonParse(row.inputs, {}),
233
+ outputs: safeJsonParse(row.outputs, {}),
234
+ digest_json: safeJsonParse(
235
+ row.digest_json,
236
+ {}
237
+ )
211
238
  }));
212
239
  } catch (error) {
213
240
  throw new DatabaseError(
@@ -249,7 +276,10 @@ class FrameDatabase {
249
276
  const createdEvent = this.db.prepare("SELECT * FROM events WHERE event_id = ?").get(event.event_id);
250
277
  return {
251
278
  ...createdEvent,
252
- payload: JSON.parse(createdEvent.payload)
279
+ payload: safeJsonParse(
280
+ createdEvent.payload,
281
+ {}
282
+ )
253
283
  };
254
284
  } catch (error) {
255
285
  throw new DatabaseError(
@@ -274,7 +304,7 @@ class FrameDatabase {
274
304
  const rows = this.db.prepare(query).all(...params);
275
305
  return rows.map((row) => ({
276
306
  ...row,
277
- payload: JSON.parse(row.payload)
307
+ payload: safeJsonParse(row.payload, {})
278
308
  }));
279
309
  } catch (error) {
280
310
  throw new DatabaseError(
@@ -332,7 +362,10 @@ class FrameDatabase {
332
362
  const createdAnchor = this.db.prepare("SELECT * FROM anchors WHERE anchor_id = ?").get(anchor.anchor_id);
333
363
  return {
334
364
  ...createdAnchor,
335
- metadata: JSON.parse(createdAnchor.metadata || "{}")
365
+ metadata: safeJsonParse(
366
+ createdAnchor.metadata,
367
+ {}
368
+ )
336
369
  };
337
370
  } catch (error) {
338
371
  throw new DatabaseError(
@@ -357,7 +390,7 @@ class FrameDatabase {
357
390
  ).all(frameId);
358
391
  return rows.map((row) => ({
359
392
  ...row,
360
- metadata: JSON.parse(row.metadata || "{}")
393
+ metadata: safeJsonParse(row.metadata, {})
361
394
  }));
362
395
  } catch (error) {
363
396
  throw new DatabaseError(