@stackmemoryai/stackmemory 0.3.14 → 0.3.16

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.
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 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}"],
5
+ "mappings": "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",
6
+ "names": []
7
+ }
@@ -127,12 +127,8 @@ class FrameDatabase {
127
127
  digest_json: JSON.parse(row.digest_json || "{}")
128
128
  };
129
129
  } catch (error) {
130
- throw new DatabaseError(
131
- `Failed to get frame: ${frameId}`,
132
- ErrorCode.DB_QUERY_FAILED,
133
- { frameId, operation: "getFrame" },
134
- error instanceof Error ? error : void 0
135
- );
130
+ logger.warn(`Failed to get frame: ${frameId}`, { error });
131
+ return void 0;
136
132
  }
137
133
  }
138
134
  /**
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/context/frame-database.ts"],
4
- "sourcesContent": ["/**\n * Frame Database Operations\n * Handles all database interactions for frames, events, and anchors\n */\n\nimport Database from 'better-sqlite3';\nimport { Frame, Event, Anchor } from './frame-types.js';\nimport { logger } from '../monitoring/logger.js';\nimport { DatabaseError, ErrorCode } from '../errors/index.js';\n\nexport class FrameDatabase {\n constructor(private db: Database.Database) {}\n\n /**\n * Initialize database schema\n */\n initSchema(): void {\n try {\n // Enable WAL mode for better concurrency\n this.db.pragma('journal_mode = WAL');\n this.db.pragma('synchronous = NORMAL');\n\n // Create frames table\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS frames (\n frame_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n parent_frame_id TEXT,\n depth INTEGER NOT NULL DEFAULT 0,\n type TEXT NOT NULL,\n name TEXT NOT NULL,\n state TEXT NOT NULL DEFAULT 'active',\n inputs TEXT DEFAULT '{}',\n outputs TEXT DEFAULT '{}',\n digest_text TEXT,\n digest_json TEXT DEFAULT '{}',\n created_at INTEGER NOT NULL DEFAULT (unixepoch()),\n closed_at INTEGER,\n FOREIGN KEY (parent_frame_id) REFERENCES frames(frame_id)\n );\n `);\n\n // Create events table\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS events (\n event_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n run_id TEXT NOT NULL,\n seq INTEGER NOT NULL,\n event_type TEXT NOT NULL,\n payload TEXT NOT NULL DEFAULT '{}',\n ts INTEGER NOT NULL DEFAULT (unixepoch() * 1000),\n FOREIGN KEY (frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );\n `);\n\n // Create anchors table\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS anchors (\n anchor_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n type TEXT NOT NULL,\n text TEXT NOT NULL,\n priority INTEGER NOT NULL DEFAULT 5,\n metadata TEXT DEFAULT '{}',\n created_at INTEGER NOT NULL DEFAULT (unixepoch()),\n FOREIGN KEY (frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );\n `);\n\n // Create indexes for performance\n this.db.exec(`\n CREATE INDEX IF NOT EXISTS idx_frames_project_state ON frames(project_id, state);\n CREATE INDEX IF NOT EXISTS idx_frames_parent ON frames(parent_frame_id);\n CREATE INDEX IF NOT EXISTS idx_events_frame_seq ON events(frame_id, seq);\n CREATE INDEX IF NOT EXISTS idx_anchors_frame_priority ON anchors(frame_id, priority DESC);\n `);\n\n logger.info('Frame database schema initialized');\n } catch (error: unknown) {\n throw new DatabaseError(\n 'Failed to initialize frame database schema',\n ErrorCode.DB_SCHEMA_ERROR,\n { operation: 'initSchema' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Insert new frame\n */\n insertFrame(frame: Omit<Frame, 'created_at' | 'closed_at'>): Frame {\n try {\n const stmt = this.db.prepare(`\n INSERT INTO frames (frame_id, run_id, project_id, parent_frame_id, depth, type, name, state, inputs, outputs, digest_json)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n `);\n\n const result = stmt.run(\n frame.frame_id,\n frame.run_id,\n frame.project_id,\n frame.parent_frame_id || null,\n frame.depth,\n frame.type,\n frame.name,\n frame.state,\n JSON.stringify(frame.inputs),\n JSON.stringify(frame.outputs),\n JSON.stringify(frame.digest_json)\n );\n\n if (result.changes === 0) {\n throw new Error('Frame insertion failed - no rows affected');\n }\n\n // Return the created frame with timestamp\n const createdFrame = this.getFrame(frame.frame_id);\n if (!createdFrame) {\n throw new Error('Failed to retrieve created frame');\n }\n\n return createdFrame;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to insert frame: ${frame.frame_id}`,\n ErrorCode.DB_INSERT_FAILED,\n {\n frameId: frame.frame_id,\n frameName: frame.name,\n operation: 'insertFrame',\n },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get frame by ID\n */\n getFrame(frameId: string): Frame | undefined {\n try {\n const row = this.db\n .prepare('SELECT * FROM frames WHERE frame_id = ?')\n .get(frameId) as any;\n\n if (!row) return undefined;\n\n return {\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n };\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get frame: ${frameId}`,\n ErrorCode.DB_QUERY_FAILED,\n { frameId, operation: 'getFrame' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Update frame state and outputs\n */\n updateFrame(frameId: string, updates: Partial<Frame>): void {\n try {\n const setClauses: string[] = [];\n const values: any[] = [];\n\n if (updates.state !== undefined) {\n setClauses.push('state = ?');\n values.push(updates.state);\n }\n\n if (updates.outputs !== undefined) {\n setClauses.push('outputs = ?');\n values.push(JSON.stringify(updates.outputs));\n }\n\n if (updates.digest_text !== undefined) {\n setClauses.push('digest_text = ?');\n values.push(updates.digest_text);\n }\n\n if (updates.digest_json !== undefined) {\n setClauses.push('digest_json = ?');\n values.push(JSON.stringify(updates.digest_json));\n }\n\n if (updates.closed_at !== undefined) {\n setClauses.push('closed_at = ?');\n values.push(updates.closed_at);\n }\n\n if (setClauses.length === 0) {\n return; // No updates to apply\n }\n\n values.push(frameId);\n\n const stmt = this.db.prepare(`\n UPDATE frames SET ${setClauses.join(', ')} WHERE frame_id = ?\n `);\n\n const result = stmt.run(...values);\n\n if (result.changes === 0) {\n throw new Error(`Frame not found: ${frameId}`);\n }\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to update frame: ${frameId}`,\n ErrorCode.DB_UPDATE_FAILED,\n { frameId, updates, operation: 'updateFrame' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get frames by project and state\n */\n getFramesByProject(projectId: string, state?: 'active' | 'closed'): Frame[] {\n try {\n const query = state\n ? 'SELECT * FROM frames WHERE project_id = ? AND state = ? ORDER BY created_at'\n : 'SELECT * FROM frames WHERE project_id = ? ORDER BY created_at';\n\n const params = state ? [projectId, state] : [projectId];\n const rows = this.db.prepare(query).all(...params) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get frames for project: ${projectId}`,\n ErrorCode.DB_QUERY_FAILED,\n { projectId, state, operation: 'getFramesByProject' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Insert event\n */\n insertEvent(event: Omit<Event, 'ts'>): Event {\n try {\n const stmt = this.db.prepare(`\n INSERT INTO events (event_id, frame_id, run_id, seq, event_type, payload)\n VALUES (?, ?, ?, ?, ?, ?)\n `);\n\n const result = stmt.run(\n event.event_id,\n event.frame_id,\n event.run_id,\n event.seq,\n event.event_type,\n JSON.stringify(event.payload)\n );\n\n if (result.changes === 0) {\n throw new Error('Event insertion failed');\n }\n\n // Return the created event with timestamp\n const createdEvent = this.db\n .prepare('SELECT * FROM events WHERE event_id = ?')\n .get(event.event_id) as any;\n\n return {\n ...createdEvent,\n payload: JSON.parse(createdEvent.payload),\n };\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to insert event: ${event.event_id}`,\n ErrorCode.DB_INSERT_FAILED,\n {\n eventId: event.event_id,\n frameId: event.frame_id,\n operation: 'insertEvent',\n },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get events for a frame\n */\n getFrameEvents(frameId: string, limit?: number): Event[] {\n try {\n const query = limit\n ? 'SELECT * FROM events WHERE frame_id = ? ORDER BY seq DESC LIMIT ?'\n : 'SELECT * FROM events WHERE frame_id = ? ORDER BY seq ASC';\n\n const params = limit ? [frameId, limit] : [frameId];\n const rows = this.db.prepare(query).all(...params) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n payload: JSON.parse(row.payload),\n }));\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get events for frame: ${frameId}`,\n ErrorCode.DB_QUERY_FAILED,\n { frameId, limit, operation: 'getFrameEvents' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get next event sequence number\n */\n getNextEventSequence(frameId: string): number {\n try {\n const result = this.db\n .prepare('SELECT MAX(seq) as max_seq FROM events WHERE frame_id = ?')\n .get(frameId) as { max_seq: number | null };\n\n return (result.max_seq || 0) + 1;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get next event sequence for frame: ${frameId}`,\n ErrorCode.DB_QUERY_FAILED,\n { frameId, operation: 'getNextEventSequence' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Insert anchor\n */\n insertAnchor(anchor: Omit<Anchor, 'created_at'>): Anchor {\n try {\n const stmt = this.db.prepare(`\n INSERT INTO anchors (anchor_id, frame_id, type, text, priority, metadata)\n VALUES (?, ?, ?, ?, ?, ?)\n `);\n\n const result = stmt.run(\n anchor.anchor_id,\n anchor.frame_id,\n anchor.type,\n anchor.text,\n anchor.priority,\n JSON.stringify(anchor.metadata)\n );\n\n if (result.changes === 0) {\n throw new Error('Anchor insertion failed');\n }\n\n // Return the created anchor with timestamp\n const createdAnchor = this.db\n .prepare('SELECT * FROM anchors WHERE anchor_id = ?')\n .get(anchor.anchor_id) as any;\n\n return {\n ...createdAnchor,\n metadata: JSON.parse(createdAnchor.metadata || '{}'),\n };\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to insert anchor: ${anchor.anchor_id}`,\n ErrorCode.DB_INSERT_FAILED,\n {\n anchorId: anchor.anchor_id,\n frameId: anchor.frame_id,\n operation: 'insertAnchor',\n },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get anchors for a frame\n */\n getFrameAnchors(frameId: string): Anchor[] {\n try {\n const rows = this.db\n .prepare(\n 'SELECT * FROM anchors WHERE frame_id = ? ORDER BY priority DESC, created_at ASC'\n )\n .all(frameId) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n }));\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get anchors for frame: ${frameId}`,\n ErrorCode.DB_QUERY_FAILED,\n { frameId, operation: 'getFrameAnchors' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Delete frame and all related data\n */\n deleteFrame(frameId: string): void {\n try {\n // Delete in order due to foreign keys\n this.db.prepare('DELETE FROM anchors WHERE frame_id = ?').run(frameId);\n this.db.prepare('DELETE FROM events WHERE frame_id = ?').run(frameId);\n this.db.prepare('DELETE FROM frames WHERE frame_id = ?').run(frameId);\n\n logger.info('Frame deleted', { frameId });\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to delete frame: ${frameId}`,\n ErrorCode.DB_DELETE_FAILED,\n { frameId, operation: 'deleteFrame' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get database statistics\n */\n getStatistics(): Record<string, number> {\n try {\n const frameCount = this.db\n .prepare('SELECT COUNT(*) as count FROM frames')\n .get() as { count: number };\n const eventCount = this.db\n .prepare('SELECT COUNT(*) as count FROM events')\n .get() as { count: number };\n const anchorCount = this.db\n .prepare('SELECT COUNT(*) as count FROM anchors')\n .get() as { count: number };\n const activeFrames = this.db\n .prepare(\"SELECT COUNT(*) as count FROM frames WHERE state = 'active'\")\n .get() as { count: number };\n\n return {\n totalFrames: frameCount.count,\n totalEvents: eventCount.count,\n totalAnchors: anchorCount.count,\n activeFrames: activeFrames.count,\n };\n } catch (error: unknown) {\n logger.warn('Failed to get database statistics', {\n error: error instanceof Error ? error.message : String(error),\n });\n return {};\n }\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/**\n * Frame Database Operations\n * Handles all database interactions for frames, events, and anchors\n */\n\nimport Database from 'better-sqlite3';\nimport { Frame, Event, Anchor } from './frame-types.js';\nimport { logger } from '../monitoring/logger.js';\nimport { DatabaseError, ErrorCode } from '../errors/index.js';\n\nexport class FrameDatabase {\n constructor(private db: Database.Database) {}\n\n /**\n * Initialize database schema\n */\n initSchema(): void {\n try {\n // Enable WAL mode for better concurrency\n this.db.pragma('journal_mode = WAL');\n this.db.pragma('synchronous = NORMAL');\n\n // Create frames table\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS frames (\n frame_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n parent_frame_id TEXT,\n depth INTEGER NOT NULL DEFAULT 0,\n type TEXT NOT NULL,\n name TEXT NOT NULL,\n state TEXT NOT NULL DEFAULT 'active',\n inputs TEXT DEFAULT '{}',\n outputs TEXT DEFAULT '{}',\n digest_text TEXT,\n digest_json TEXT DEFAULT '{}',\n created_at INTEGER NOT NULL DEFAULT (unixepoch()),\n closed_at INTEGER,\n FOREIGN KEY (parent_frame_id) REFERENCES frames(frame_id)\n );\n `);\n\n // Create events table\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS events (\n event_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n run_id TEXT NOT NULL,\n seq INTEGER NOT NULL,\n event_type TEXT NOT NULL,\n payload TEXT NOT NULL DEFAULT '{}',\n ts INTEGER NOT NULL DEFAULT (unixepoch() * 1000),\n FOREIGN KEY (frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );\n `);\n\n // Create anchors table\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS anchors (\n anchor_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n type TEXT NOT NULL,\n text TEXT NOT NULL,\n priority INTEGER NOT NULL DEFAULT 5,\n metadata TEXT DEFAULT '{}',\n created_at INTEGER NOT NULL DEFAULT (unixepoch()),\n FOREIGN KEY (frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );\n `);\n\n // Create indexes for performance\n this.db.exec(`\n CREATE INDEX IF NOT EXISTS idx_frames_project_state ON frames(project_id, state);\n CREATE INDEX IF NOT EXISTS idx_frames_parent ON frames(parent_frame_id);\n CREATE INDEX IF NOT EXISTS idx_events_frame_seq ON events(frame_id, seq);\n CREATE INDEX IF NOT EXISTS idx_anchors_frame_priority ON anchors(frame_id, priority DESC);\n `);\n\n logger.info('Frame database schema initialized');\n } catch (error: unknown) {\n throw new DatabaseError(\n 'Failed to initialize frame database schema',\n ErrorCode.DB_SCHEMA_ERROR,\n { operation: 'initSchema' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Insert new frame\n */\n insertFrame(frame: Omit<Frame, 'created_at' | 'closed_at'>): Frame {\n try {\n const stmt = this.db.prepare(`\n INSERT INTO frames (frame_id, run_id, project_id, parent_frame_id, depth, type, name, state, inputs, outputs, digest_json)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n `);\n\n const result = stmt.run(\n frame.frame_id,\n frame.run_id,\n frame.project_id,\n frame.parent_frame_id || null,\n frame.depth,\n frame.type,\n frame.name,\n frame.state,\n JSON.stringify(frame.inputs),\n JSON.stringify(frame.outputs),\n JSON.stringify(frame.digest_json)\n );\n\n if (result.changes === 0) {\n throw new Error('Frame insertion failed - no rows affected');\n }\n\n // Return the created frame with timestamp\n const createdFrame = this.getFrame(frame.frame_id);\n if (!createdFrame) {\n throw new Error('Failed to retrieve created frame');\n }\n\n return createdFrame;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to insert frame: ${frame.frame_id}`,\n ErrorCode.DB_INSERT_FAILED,\n {\n frameId: frame.frame_id,\n frameName: frame.name,\n operation: 'insertFrame',\n },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get frame by ID\n */\n getFrame(frameId: string): Frame | undefined {\n try {\n const row = this.db\n .prepare('SELECT * FROM frames WHERE frame_id = ?')\n .get(frameId) as any;\n\n if (!row) return undefined;\n\n return {\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n };\n } catch (error: unknown) {\n logger.warn(`Failed to get frame: ${frameId}`, { error });\n return undefined;\n }\n }\n\n /**\n * Update frame state and outputs\n */\n updateFrame(frameId: string, updates: Partial<Frame>): void {\n try {\n const setClauses: string[] = [];\n const values: any[] = [];\n\n if (updates.state !== undefined) {\n setClauses.push('state = ?');\n values.push(updates.state);\n }\n\n if (updates.outputs !== undefined) {\n setClauses.push('outputs = ?');\n values.push(JSON.stringify(updates.outputs));\n }\n\n if (updates.digest_text !== undefined) {\n setClauses.push('digest_text = ?');\n values.push(updates.digest_text);\n }\n\n if (updates.digest_json !== undefined) {\n setClauses.push('digest_json = ?');\n values.push(JSON.stringify(updates.digest_json));\n }\n\n if (updates.closed_at !== undefined) {\n setClauses.push('closed_at = ?');\n values.push(updates.closed_at);\n }\n\n if (setClauses.length === 0) {\n return; // No updates to apply\n }\n\n values.push(frameId);\n\n const stmt = this.db.prepare(`\n UPDATE frames SET ${setClauses.join(', ')} WHERE frame_id = ?\n `);\n\n const result = stmt.run(...values);\n\n if (result.changes === 0) {\n throw new Error(`Frame not found: ${frameId}`);\n }\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to update frame: ${frameId}`,\n ErrorCode.DB_UPDATE_FAILED,\n { frameId, updates, operation: 'updateFrame' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get frames by project and state\n */\n getFramesByProject(projectId: string, state?: 'active' | 'closed'): Frame[] {\n try {\n const query = state\n ? 'SELECT * FROM frames WHERE project_id = ? AND state = ? ORDER BY created_at'\n : 'SELECT * FROM frames WHERE project_id = ? ORDER BY created_at';\n\n const params = state ? [projectId, state] : [projectId];\n const rows = this.db.prepare(query).all(...params) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get frames for project: ${projectId}`,\n ErrorCode.DB_QUERY_FAILED,\n { projectId, state, operation: 'getFramesByProject' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Insert event\n */\n insertEvent(event: Omit<Event, 'ts'>): Event {\n try {\n const stmt = this.db.prepare(`\n INSERT INTO events (event_id, frame_id, run_id, seq, event_type, payload)\n VALUES (?, ?, ?, ?, ?, ?)\n `);\n\n const result = stmt.run(\n event.event_id,\n event.frame_id,\n event.run_id,\n event.seq,\n event.event_type,\n JSON.stringify(event.payload)\n );\n\n if (result.changes === 0) {\n throw new Error('Event insertion failed');\n }\n\n // Return the created event with timestamp\n const createdEvent = this.db\n .prepare('SELECT * FROM events WHERE event_id = ?')\n .get(event.event_id) as any;\n\n return {\n ...createdEvent,\n payload: JSON.parse(createdEvent.payload),\n };\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to insert event: ${event.event_id}`,\n ErrorCode.DB_INSERT_FAILED,\n {\n eventId: event.event_id,\n frameId: event.frame_id,\n operation: 'insertEvent',\n },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get events for a frame\n */\n getFrameEvents(frameId: string, limit?: number): Event[] {\n try {\n const query = limit\n ? 'SELECT * FROM events WHERE frame_id = ? ORDER BY seq DESC LIMIT ?'\n : 'SELECT * FROM events WHERE frame_id = ? ORDER BY seq ASC';\n\n const params = limit ? [frameId, limit] : [frameId];\n const rows = this.db.prepare(query).all(...params) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n payload: JSON.parse(row.payload),\n }));\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get events for frame: ${frameId}`,\n ErrorCode.DB_QUERY_FAILED,\n { frameId, limit, operation: 'getFrameEvents' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get next event sequence number\n */\n getNextEventSequence(frameId: string): number {\n try {\n const result = this.db\n .prepare('SELECT MAX(seq) as max_seq FROM events WHERE frame_id = ?')\n .get(frameId) as { max_seq: number | null };\n\n return (result.max_seq || 0) + 1;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get next event sequence for frame: ${frameId}`,\n ErrorCode.DB_QUERY_FAILED,\n { frameId, operation: 'getNextEventSequence' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Insert anchor\n */\n insertAnchor(anchor: Omit<Anchor, 'created_at'>): Anchor {\n try {\n const stmt = this.db.prepare(`\n INSERT INTO anchors (anchor_id, frame_id, type, text, priority, metadata)\n VALUES (?, ?, ?, ?, ?, ?)\n `);\n\n const result = stmt.run(\n anchor.anchor_id,\n anchor.frame_id,\n anchor.type,\n anchor.text,\n anchor.priority,\n JSON.stringify(anchor.metadata)\n );\n\n if (result.changes === 0) {\n throw new Error('Anchor insertion failed');\n }\n\n // Return the created anchor with timestamp\n const createdAnchor = this.db\n .prepare('SELECT * FROM anchors WHERE anchor_id = ?')\n .get(anchor.anchor_id) as any;\n\n return {\n ...createdAnchor,\n metadata: JSON.parse(createdAnchor.metadata || '{}'),\n };\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to insert anchor: ${anchor.anchor_id}`,\n ErrorCode.DB_INSERT_FAILED,\n {\n anchorId: anchor.anchor_id,\n frameId: anchor.frame_id,\n operation: 'insertAnchor',\n },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get anchors for a frame\n */\n getFrameAnchors(frameId: string): Anchor[] {\n try {\n const rows = this.db\n .prepare(\n 'SELECT * FROM anchors WHERE frame_id = ? ORDER BY priority DESC, created_at ASC'\n )\n .all(frameId) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n }));\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get anchors for frame: ${frameId}`,\n ErrorCode.DB_QUERY_FAILED,\n { frameId, operation: 'getFrameAnchors' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Delete frame and all related data\n */\n deleteFrame(frameId: string): void {\n try {\n // Delete in order due to foreign keys\n this.db.prepare('DELETE FROM anchors WHERE frame_id = ?').run(frameId);\n this.db.prepare('DELETE FROM events WHERE frame_id = ?').run(frameId);\n this.db.prepare('DELETE FROM frames WHERE frame_id = ?').run(frameId);\n\n logger.info('Frame deleted', { frameId });\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to delete frame: ${frameId}`,\n ErrorCode.DB_DELETE_FAILED,\n { frameId, operation: 'deleteFrame' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get database statistics\n */\n getStatistics(): Record<string, number> {\n try {\n const frameCount = this.db\n .prepare('SELECT COUNT(*) as count FROM frames')\n .get() as { count: number };\n const eventCount = this.db\n .prepare('SELECT COUNT(*) as count FROM events')\n .get() as { count: number };\n const anchorCount = this.db\n .prepare('SELECT COUNT(*) as count FROM anchors')\n .get() as { count: number };\n const activeFrames = this.db\n .prepare(\"SELECT COUNT(*) as count FROM frames WHERE state = 'active'\")\n .get() as { count: number };\n\n return {\n totalFrames: frameCount.count,\n totalEvents: eventCount.count,\n totalAnchors: anchorCount.count,\n activeFrames: activeFrames.count,\n };\n } catch (error: unknown) {\n logger.warn('Failed to get database statistics', {\n error: error instanceof Error ? error.message : String(error),\n });\n return {};\n }\n }\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }