@stackmemoryai/stackmemory 0.5.33 → 0.5.34

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 (107) 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 +8 -0
  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 +29 -0
  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 +140 -34
  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 +4 -0
  79. package/dist/hooks/sms-action-runner.js.map +2 -2
  80. package/dist/hooks/whatsapp-commands.js +142 -2
  81. package/dist/hooks/whatsapp-commands.js.map +2 -2
  82. package/dist/hooks/whatsapp-sync.js +34 -0
  83. package/dist/hooks/whatsapp-sync.js.map +2 -2
  84. package/dist/index.js +1 -1
  85. package/dist/index.js.map +1 -1
  86. package/dist/integrations/mcp/handlers/context-handlers.js.map +1 -1
  87. package/dist/integrations/mcp/handlers/discovery-handlers.js.map +1 -1
  88. package/dist/integrations/mcp/server.js +1 -1
  89. package/dist/integrations/mcp/server.js.map +1 -1
  90. package/dist/integrations/ralph/bridge/ralph-stackmemory-bridge.js +1 -1
  91. package/dist/integrations/ralph/bridge/ralph-stackmemory-bridge.js.map +1 -1
  92. package/dist/integrations/ralph/context/stackmemory-context-loader.js +1 -1
  93. package/dist/integrations/ralph/context/stackmemory-context-loader.js.map +1 -1
  94. package/dist/integrations/ralph/learning/pattern-learner.js +1 -1
  95. package/dist/integrations/ralph/learning/pattern-learner.js.map +1 -1
  96. package/dist/integrations/ralph/orchestration/multi-loop-orchestrator.js +1 -1
  97. package/dist/integrations/ralph/orchestration/multi-loop-orchestrator.js.map +1 -1
  98. package/dist/integrations/ralph/swarm/swarm-coordinator.js +1 -1
  99. package/dist/integrations/ralph/swarm/swarm-coordinator.js.map +1 -1
  100. package/dist/integrations/ralph/visualization/ralph-debugger.js +1 -1
  101. package/dist/integrations/ralph/visualization/ralph-debugger.js.map +1 -1
  102. package/dist/mcp/stackmemory-mcp-server.js +1 -1
  103. package/dist/mcp/stackmemory-mcp-server.js.map +1 -1
  104. package/dist/skills/claude-skills.js.map +1 -1
  105. package/dist/skills/recursive-agent-orchestrator.js.map +1 -1
  106. package/dist/skills/unified-rlm-orchestrator.js.map +1 -1
  107. package/package.json +1 -1
@@ -2,281 +2,48 @@ import { fileURLToPath as __fileURLToPath } from 'url';
2
2
  import { dirname as __pathDirname } from 'path';
3
3
  const __filename = __fileURLToPath(import.meta.url);
4
4
  const __dirname = __pathDirname(__filename);
5
- import { logger } from "./logger.js";
6
- var ErrorCode = /* @__PURE__ */ ((ErrorCode2) => {
7
- ErrorCode2["AUTH_FAILED"] = "AUTH_FAILED";
8
- ErrorCode2["TOKEN_EXPIRED"] = "TOKEN_EXPIRED";
9
- ErrorCode2["INVALID_CREDENTIALS"] = "INVALID_CREDENTIALS";
10
- ErrorCode2["FILE_NOT_FOUND"] = "FILE_NOT_FOUND";
11
- ErrorCode2["PERMISSION_DENIED"] = "PERMISSION_DENIED";
12
- ErrorCode2["DISK_FULL"] = "DISK_FULL";
13
- ErrorCode2["NOT_GIT_REPO"] = "NOT_GIT_REPO";
14
- ErrorCode2["GIT_COMMAND_FAILED"] = "GIT_COMMAND_FAILED";
15
- ErrorCode2["INVALID_BRANCH"] = "INVALID_BRANCH";
16
- ErrorCode2["DB_CONNECTION_FAILED"] = "DB_CONNECTION_FAILED";
17
- ErrorCode2["DB_QUERY_FAILED"] = "DB_QUERY_FAILED";
18
- ErrorCode2["DB_CORRUPTION"] = "DB_CORRUPTION";
19
- ErrorCode2["NETWORK_ERROR"] = "NETWORK_ERROR";
20
- ErrorCode2["API_ERROR"] = "API_ERROR";
21
- ErrorCode2["TIMEOUT"] = "TIMEOUT";
22
- ErrorCode2["INVALID_INPUT"] = "INVALID_INPUT";
23
- ErrorCode2["VALIDATION_FAILED"] = "VALIDATION_FAILED";
24
- ErrorCode2["UNKNOWN_ERROR"] = "UNKNOWN_ERROR";
25
- ErrorCode2["OPERATION_FAILED"] = "OPERATION_FAILED";
26
- ErrorCode2["CONFIGURATION_ERROR"] = "CONFIGURATION_ERROR";
27
- return ErrorCode2;
28
- })(ErrorCode || {});
29
- class StackMemoryError extends Error {
30
- code;
31
- context;
32
- userMessage;
33
- recoverable;
34
- constructor(code, message, userMessage, context = {}, recoverable = false, cause) {
35
- super(message);
36
- this.name = "StackMemoryError";
37
- this.code = code;
38
- this.context = context;
39
- this.userMessage = userMessage || this.getDefaultUserMessage(code);
40
- this.recoverable = recoverable;
41
- if (cause && Error.captureStackTrace) {
42
- Error.captureStackTrace(this, StackMemoryError);
43
- }
44
- logger.error(message, cause, {
45
- code,
46
- context,
47
- recoverable,
48
- userMessage: this.userMessage
49
- });
50
- }
51
- getDefaultUserMessage(code) {
52
- switch (code) {
53
- case "AUTH_FAILED" /* AUTH_FAILED */:
54
- return "Authentication failed. Please check your credentials and try again.";
55
- case "NOT_GIT_REPO" /* NOT_GIT_REPO */:
56
- return "This command requires a git repository. Please run it from within a git repository.";
57
- case "PERMISSION_DENIED" /* PERMISSION_DENIED */:
58
- return "Permission denied. Please check file permissions or run with appropriate privileges.";
59
- case "NETWORK_ERROR" /* NETWORK_ERROR */:
60
- return "Network error. Please check your internet connection and try again.";
61
- case "INVALID_INPUT" /* INVALID_INPUT */:
62
- return "Invalid input provided. Please check your command and try again.";
63
- case "DB_CONNECTION_FAILED" /* DB_CONNECTION_FAILED */:
64
- return "Database connection failed. Please try again or contact support if the issue persists.";
65
- case "GIT_COMMAND_FAILED" /* GIT_COMMAND_FAILED */:
66
- return "Git operation failed. Please ensure your repository is in a valid state.";
67
- default:
68
- return "An unexpected error occurred. Please try again or contact support.";
69
- }
70
- }
71
- static fromNodeError(nodeError, context = {}) {
72
- const code = nodeError.code;
73
- switch (code) {
74
- case "ENOENT":
75
- return new StackMemoryError(
76
- "FILE_NOT_FOUND" /* FILE_NOT_FOUND */,
77
- `File or directory not found: ${nodeError.path}`,
78
- "The requested file or directory was not found.",
79
- { ...context, path: nodeError.path },
80
- false,
81
- nodeError
82
- );
83
- case "EACCES":
84
- case "EPERM":
85
- return new StackMemoryError(
86
- "PERMISSION_DENIED" /* PERMISSION_DENIED */,
87
- `Permission denied: ${nodeError.path}`,
88
- "Permission denied. Please check file permissions.",
89
- { ...context, path: nodeError.path },
90
- true,
91
- nodeError
92
- );
93
- case "ENOSPC":
94
- return new StackMemoryError(
95
- "DISK_FULL" /* DISK_FULL */,
96
- "No space left on device",
97
- "Insufficient disk space. Please free up space and try again.",
98
- context,
99
- true,
100
- nodeError
101
- );
102
- case "ETIMEDOUT":
103
- return new StackMemoryError(
104
- "TIMEOUT" /* TIMEOUT */,
105
- "Operation timed out",
106
- "The operation timed out. Please try again.",
107
- context,
108
- true,
109
- nodeError
110
- );
111
- default:
112
- return new StackMemoryError(
113
- "UNKNOWN_ERROR" /* UNKNOWN_ERROR */,
114
- nodeError.message,
115
- "An unexpected system error occurred.",
116
- { ...context, nodeErrorCode: code },
117
- false,
118
- nodeError
119
- );
120
- }
121
- }
122
- }
123
- class ErrorHandler {
124
- static retryMap = /* @__PURE__ */ new Map();
125
- static MAX_RETRIES = 3;
126
- static handle(error, operation) {
127
- if (error instanceof StackMemoryError) {
128
- console.error(`\u274C ${error.userMessage}`);
129
- if (error.recoverable) {
130
- console.error("\u{1F4A1} This error may be recoverable. Please try again.");
131
- }
132
- process.exit(1);
133
- }
134
- if (error instanceof Error) {
135
- let stackMemoryError;
136
- if ("code" in error && typeof error.code === "string") {
137
- stackMemoryError = StackMemoryError.fromNodeError(
138
- error,
139
- { operation }
140
- );
141
- } else {
142
- stackMemoryError = new StackMemoryError(
143
- "OPERATION_FAILED" /* OPERATION_FAILED */,
144
- `Operation '${operation}' failed: ${error.message}`,
145
- `Operation failed: ${error.message}`,
146
- { operation },
147
- false,
148
- error
149
- );
150
- }
151
- console.error(`\u274C ${stackMemoryError.userMessage}`);
152
- if (stackMemoryError.recoverable) {
153
- console.error("\u{1F4A1} This error may be recoverable. Please try again.");
154
- }
155
- process.exit(1);
156
- }
157
- const unknownError = new StackMemoryError(
158
- "UNKNOWN_ERROR" /* UNKNOWN_ERROR */,
159
- `Unknown error in operation '${operation}': ${String(error)}`,
160
- "An unexpected error occurred.",
161
- { operation, errorType: typeof error },
162
- false
163
- );
164
- console.error(`\u274C ${unknownError.userMessage}`);
165
- process.exit(1);
166
- }
167
- static async safeExecute(operation, operationName, fallback) {
168
- try {
169
- return await operation();
170
- } catch (error) {
171
- if (fallback !== void 0) {
172
- logger.warn(`Operation '${operationName}' failed, using fallback`, {
173
- error: String(error)
174
- });
175
- return fallback;
176
- }
177
- ErrorHandler.handle(error, operationName);
178
- }
179
- }
180
- static async withRetry(operation, operationName, maxRetries = ErrorHandler.MAX_RETRIES) {
181
- let lastError;
182
- for (let attempt = 1; attempt <= maxRetries; attempt++) {
183
- try {
184
- const result = await operation();
185
- ErrorHandler.retryMap.delete(operationName);
186
- return result;
187
- } catch (error) {
188
- lastError = error;
189
- if (error instanceof StackMemoryError && !error.recoverable) {
190
- ErrorHandler.handle(error, operationName);
191
- }
192
- if (attempt === maxRetries) {
193
- break;
194
- }
195
- const delay = Math.min(1e3 * Math.pow(2, attempt - 1), 5e3);
196
- logger.warn(
197
- `Attempt ${attempt}/${maxRetries} failed for '${operationName}', retrying in ${delay}ms`,
198
- {
199
- error: String(error)
200
- }
201
- );
202
- await new Promise((resolve) => setTimeout(resolve, delay));
203
- }
204
- }
205
- ErrorHandler.handle(
206
- lastError,
207
- `${operationName} (after ${maxRetries} attempts)`
208
- );
209
- }
210
- static createCircuitBreaker(operation, operationName, threshold = 5) {
211
- let failures = 0;
212
- let lastFailure = 0;
213
- const resetTimeout = 3e4;
214
- return async () => {
215
- const now = Date.now();
216
- if (now - lastFailure > resetTimeout) {
217
- failures = 0;
218
- }
219
- if (failures >= threshold) {
220
- throw new StackMemoryError(
221
- "OPERATION_FAILED" /* OPERATION_FAILED */,
222
- `Circuit breaker open for '${operationName}'`,
223
- `Operation temporarily unavailable. Please try again later.`,
224
- { operationName, failures, threshold },
225
- true
226
- );
227
- }
228
- try {
229
- const result = await operation();
230
- failures = 0;
231
- return result;
232
- } catch (error) {
233
- failures++;
234
- lastFailure = now;
235
- throw error;
236
- }
237
- };
238
- }
239
- }
240
- const validateInput = (value, name, validator) => {
241
- if (!validator(value)) {
242
- throw new StackMemoryError(
243
- "INVALID_INPUT" /* INVALID_INPUT */,
244
- `Invalid ${name}: ${String(value)}`,
245
- `Please provide a valid ${name}.`,
246
- { name, value },
247
- true
248
- );
249
- }
250
- };
251
- const validateEmail = (email) => {
252
- const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
253
- if (!emailRegex.test(email) || email.length > 254) {
254
- throw new StackMemoryError(
255
- "INVALID_INPUT" /* INVALID_INPUT */,
256
- `Invalid email format: ${email}`,
257
- "Please provide a valid email address.",
258
- { email },
259
- true
260
- );
261
- }
262
- };
263
- const validatePath = (filePath) => {
264
- if (!filePath || filePath.includes("..") || filePath.includes("\0")) {
265
- throw new StackMemoryError(
266
- "INVALID_INPUT" /* INVALID_INPUT */,
267
- `Invalid path: ${filePath}`,
268
- "Invalid file path provided.",
269
- { path: filePath },
270
- true
271
- );
272
- }
273
- };
5
+ import {
6
+ ErrorCode,
7
+ StackMemoryError,
8
+ DatabaseError,
9
+ FrameError,
10
+ TaskError,
11
+ IntegrationError,
12
+ MCPError,
13
+ ValidationError,
14
+ ProjectError,
15
+ SystemError,
16
+ ErrorHandler,
17
+ getUserFriendlyMessage,
18
+ isRetryableError,
19
+ getErrorMessage,
20
+ wrapError,
21
+ isStackMemoryError,
22
+ createErrorHandler,
23
+ validateInput,
24
+ validateEmail,
25
+ validatePath
26
+ } from "../errors/index.js";
274
27
  export {
28
+ DatabaseError,
275
29
  ErrorCode,
276
30
  ErrorHandler,
31
+ FrameError,
32
+ IntegrationError,
33
+ MCPError,
34
+ ProjectError,
277
35
  StackMemoryError,
36
+ SystemError,
37
+ TaskError,
38
+ ValidationError,
39
+ createErrorHandler,
40
+ getErrorMessage,
41
+ getUserFriendlyMessage,
42
+ isRetryableError,
43
+ isStackMemoryError,
278
44
  validateEmail,
279
45
  validateInput,
280
- validatePath
46
+ validatePath,
47
+ wrapError
281
48
  };
282
49
  //# sourceMappingURL=error-handler.js.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/monitoring/error-handler.ts"],
4
- "sourcesContent": ["/**\n * Comprehensive error handling for StackMemory CLI\n */\n\nimport { logger } from './logger.js';\n\nexport enum ErrorCode {\n // Authentication errors\n AUTH_FAILED = 'AUTH_FAILED',\n TOKEN_EXPIRED = 'TOKEN_EXPIRED',\n INVALID_CREDENTIALS = 'INVALID_CREDENTIALS',\n\n // File system errors\n FILE_NOT_FOUND = 'FILE_NOT_FOUND',\n PERMISSION_DENIED = 'PERMISSION_DENIED',\n DISK_FULL = 'DISK_FULL',\n\n // Git operation errors\n NOT_GIT_REPO = 'NOT_GIT_REPO',\n GIT_COMMAND_FAILED = 'GIT_COMMAND_FAILED',\n INVALID_BRANCH = 'INVALID_BRANCH',\n\n // Database errors\n DB_CONNECTION_FAILED = 'DB_CONNECTION_FAILED',\n DB_QUERY_FAILED = 'DB_QUERY_FAILED',\n DB_CORRUPTION = 'DB_CORRUPTION',\n\n // Network errors\n NETWORK_ERROR = 'NETWORK_ERROR',\n API_ERROR = 'API_ERROR',\n TIMEOUT = 'TIMEOUT',\n\n // Validation errors\n INVALID_INPUT = 'INVALID_INPUT',\n VALIDATION_FAILED = 'VALIDATION_FAILED',\n\n // General errors\n UNKNOWN_ERROR = 'UNKNOWN_ERROR',\n OPERATION_FAILED = 'OPERATION_FAILED',\n CONFIGURATION_ERROR = 'CONFIGURATION_ERROR',\n}\n\nexport class StackMemoryError extends Error {\n public readonly code: ErrorCode;\n public readonly context: Record<string, unknown>;\n public readonly userMessage: string;\n public readonly recoverable: boolean;\n\n constructor(\n code: ErrorCode,\n message: string,\n userMessage?: string,\n context: Record<string, unknown> = {},\n recoverable: boolean = false,\n cause?: Error\n ) {\n super(message);\n this.name = 'StackMemoryError';\n this.code = code;\n this.context = context;\n this.userMessage = userMessage || this.getDefaultUserMessage(code);\n this.recoverable = recoverable;\n\n if (cause && Error.captureStackTrace) {\n Error.captureStackTrace(this, StackMemoryError);\n }\n\n // Log the error\n logger.error(message, cause, {\n code,\n context,\n recoverable,\n userMessage: this.userMessage,\n });\n }\n\n private getDefaultUserMessage(code: ErrorCode): string {\n switch (code) {\n case ErrorCode.AUTH_FAILED:\n return 'Authentication failed. Please check your credentials and try again.';\n case ErrorCode.NOT_GIT_REPO:\n return 'This command requires a git repository. Please run it from within a git repository.';\n case ErrorCode.PERMISSION_DENIED:\n return 'Permission denied. Please check file permissions or run with appropriate privileges.';\n case ErrorCode.NETWORK_ERROR:\n return 'Network error. Please check your internet connection and try again.';\n case ErrorCode.INVALID_INPUT:\n return 'Invalid input provided. Please check your command and try again.';\n case ErrorCode.DB_CONNECTION_FAILED:\n return 'Database connection failed. Please try again or contact support if the issue persists.';\n case ErrorCode.GIT_COMMAND_FAILED:\n return 'Git operation failed. Please ensure your repository is in a valid state.';\n default:\n return 'An unexpected error occurred. Please try again or contact support.';\n }\n }\n\n static fromNodeError(\n nodeError: NodeJS.ErrnoException,\n context: Record<string, unknown> = {}\n ): StackMemoryError {\n const code = nodeError.code;\n\n switch (code) {\n case 'ENOENT':\n return new StackMemoryError(\n ErrorCode.FILE_NOT_FOUND,\n `File or directory not found: ${nodeError.path}`,\n 'The requested file or directory was not found.',\n { ...context, path: nodeError.path },\n false,\n nodeError\n );\n\n case 'EACCES':\n case 'EPERM':\n return new StackMemoryError(\n ErrorCode.PERMISSION_DENIED,\n `Permission denied: ${nodeError.path}`,\n 'Permission denied. Please check file permissions.',\n { ...context, path: nodeError.path },\n true,\n nodeError\n );\n\n case 'ENOSPC':\n return new StackMemoryError(\n ErrorCode.DISK_FULL,\n 'No space left on device',\n 'Insufficient disk space. Please free up space and try again.',\n context,\n true,\n nodeError\n );\n\n case 'ETIMEDOUT':\n return new StackMemoryError(\n ErrorCode.TIMEOUT,\n 'Operation timed out',\n 'The operation timed out. Please try again.',\n context,\n true,\n nodeError\n );\n\n default:\n return new StackMemoryError(\n ErrorCode.UNKNOWN_ERROR,\n nodeError.message,\n 'An unexpected system error occurred.',\n { ...context, nodeErrorCode: code },\n false,\n nodeError\n );\n }\n }\n}\n\nexport class ErrorHandler {\n private static retryMap = new Map<string, number>();\n private static readonly MAX_RETRIES = 3;\n\n static handle(error: unknown, operation: string): never {\n if (error instanceof StackMemoryError) {\n // Already a well-formed StackMemory error\n console.error(`\u274C ${error.userMessage}`);\n\n if (error.recoverable) {\n console.error('\uD83D\uDCA1 This error may be recoverable. Please try again.');\n }\n\n process.exit(1);\n }\n\n if (error instanceof Error) {\n // Convert Node.js error to StackMemoryError\n let stackMemoryError: StackMemoryError;\n\n if ('code' in error && typeof error.code === 'string') {\n stackMemoryError = StackMemoryError.fromNodeError(\n error as NodeJS.ErrnoException,\n { operation }\n );\n } else {\n stackMemoryError = new StackMemoryError(\n ErrorCode.OPERATION_FAILED,\n `Operation '${operation}' failed: ${error.message}`,\n `Operation failed: ${error.message}`,\n { operation },\n false,\n error\n );\n }\n\n console.error(`\u274C ${stackMemoryError.userMessage}`);\n if (stackMemoryError.recoverable) {\n console.error('\uD83D\uDCA1 This error may be recoverable. Please try again.');\n }\n\n process.exit(1);\n }\n\n // Unknown error type\n const unknownError = new StackMemoryError(\n ErrorCode.UNKNOWN_ERROR,\n `Unknown error in operation '${operation}': ${String(error)}`,\n 'An unexpected error occurred.',\n { operation, errorType: typeof error },\n false\n );\n\n console.error(`\u274C ${unknownError.userMessage}`);\n process.exit(1);\n }\n\n static async safeExecute<T>(\n operation: () => Promise<T> | T,\n operationName: string,\n fallback?: T\n ): Promise<T | undefined> {\n try {\n return await operation();\n } catch (error: unknown) {\n if (fallback !== undefined) {\n logger.warn(`Operation '${operationName}' failed, using fallback`, {\n error: String(error),\n });\n return fallback;\n }\n\n ErrorHandler.handle(error, operationName);\n }\n }\n\n static async withRetry<T>(\n operation: () => Promise<T> | T,\n operationName: string,\n maxRetries: number = ErrorHandler.MAX_RETRIES\n ): Promise<T> {\n let lastError: unknown;\n\n for (let attempt = 1; attempt <= maxRetries; attempt++) {\n try {\n const result = await operation();\n // Clear retry count on success\n ErrorHandler.retryMap.delete(operationName);\n return result;\n } catch (error: unknown) {\n lastError = error;\n\n if (error instanceof StackMemoryError && !error.recoverable) {\n // Don't retry non-recoverable errors\n ErrorHandler.handle(error, operationName);\n }\n\n if (attempt === maxRetries) {\n break;\n }\n\n const delay = Math.min(1000 * Math.pow(2, attempt - 1), 5000); // Exponential backoff\n logger.warn(\n `Attempt ${attempt}/${maxRetries} failed for '${operationName}', retrying in ${delay}ms`,\n {\n error: String(error),\n }\n );\n\n await new Promise((resolve) => setTimeout(resolve, delay));\n }\n }\n\n ErrorHandler.handle(\n lastError,\n `${operationName} (after ${maxRetries} attempts)`\n );\n }\n\n static createCircuitBreaker<T>(\n operation: () => Promise<T> | T,\n operationName: string,\n threshold: number = 5\n ) {\n let failures = 0;\n let lastFailure = 0;\n const resetTimeout = 30000; // 30 seconds\n\n return async (): Promise<T> => {\n const now = Date.now();\n\n // Reset circuit breaker after timeout\n if (now - lastFailure > resetTimeout) {\n failures = 0;\n }\n\n // Circuit is open (too many failures)\n if (failures >= threshold) {\n throw new StackMemoryError(\n ErrorCode.OPERATION_FAILED,\n `Circuit breaker open for '${operationName}'`,\n `Operation temporarily unavailable. Please try again later.`,\n { operationName, failures, threshold },\n true\n );\n }\n\n try {\n const result = await operation();\n failures = 0; // Reset on success\n return result;\n } catch (error: unknown) {\n failures++;\n lastFailure = now;\n throw error;\n }\n };\n }\n}\n\n// Utility functions for common error scenarios\nexport const validateInput = (\n value: unknown,\n name: string,\n validator: (val: unknown) => boolean\n): asserts value is NonNullable<unknown> => {\n if (!validator(value)) {\n throw new StackMemoryError(\n ErrorCode.INVALID_INPUT,\n `Invalid ${name}: ${String(value)}`,\n `Please provide a valid ${name}.`,\n { name, value },\n true\n );\n }\n};\n\nexport const validateEmail = (email: string): asserts email is string => {\n const emailRegex = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/;\n if (!emailRegex.test(email) || email.length > 254) {\n throw new StackMemoryError(\n ErrorCode.INVALID_INPUT,\n `Invalid email format: ${email}`,\n 'Please provide a valid email address.',\n { email },\n true\n );\n }\n};\n\nexport const validatePath = (filePath: string): asserts filePath is string => {\n if (!filePath || filePath.includes('..') || filePath.includes('\\0')) {\n throw new StackMemoryError(\n ErrorCode.INVALID_INPUT,\n `Invalid path: ${filePath}`,\n 'Invalid file path provided.',\n { path: filePath },\n true\n );\n }\n};\n"],
5
- "mappings": "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",
6
- "names": ["ErrorCode"]
4
+ "sourcesContent": ["/**\n * Error handling re-exports for backwards compatibility\n *\n * All error handling is now consolidated in src/core/errors/index.ts\n * This module re-exports for code that imports from monitoring/error-handler\n */\n\nexport {\n // Error codes\n ErrorCode,\n\n // Error classes\n StackMemoryError,\n DatabaseError,\n FrameError,\n TaskError,\n IntegrationError,\n MCPError,\n ValidationError,\n ProjectError,\n SystemError,\n\n // Error handler\n ErrorHandler,\n\n // Utilities\n getUserFriendlyMessage,\n isRetryableError,\n getErrorMessage,\n wrapError,\n isStackMemoryError,\n createErrorHandler,\n\n // Validators\n validateInput,\n validateEmail,\n validatePath,\n\n // Types\n type ErrorContext,\n type StackMemoryErrorOptions,\n} from '../errors/index.js';\n"],
5
+ "mappings": ";;;;AAOA;AAAA,EAEE;AAAA,EAGA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EAGA;AAAA,EAGA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EAGA;AAAA,EACA;AAAA,EACA;AAAA,OAKK;",
6
+ "names": []
7
7
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/monitoring/session-monitor.ts"],
4
- "sourcesContent": ["/**\n * Session Monitor for StackMemory\n * Automatically triggers features based on session events\n */\n\nimport { EventEmitter } from 'events';\nimport { ClearSurvival } from '../session/clear-survival.js';\nimport { HandoffGenerator } from '../session/handoff-generator.js';\nimport { FrameManager } from '../frame/frame-manager.js';\nimport { DatabaseManager } from '../storage/database-manager.js';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\n\nexport interface MonitorConfig {\n // Context thresholds\n contextWarningThreshold: number; // Default 0.6 (60%)\n contextCriticalThreshold: number; // Default 0.7 (70%)\n contextAutoSaveThreshold: number; // Default 0.85 (85%)\n\n // Handoff triggers\n idleTimeoutMinutes: number; // Default 5 minutes\n sessionEndHandoff: boolean; // Default true\n\n // Monitoring intervals\n checkIntervalSeconds: number; // Default 30 seconds\n\n // Auto actions\n autoSaveLedger: boolean; // Default true\n autoGenerateHandoff: boolean; // Default true\n autoCompactOnThreshold: boolean; // Default false\n}\n\nexport class SessionMonitor extends EventEmitter {\n private config: MonitorConfig;\n private clearSurvival: ClearSurvival;\n private handoffGenerator: HandoffGenerator;\n private frameManager: FrameManager;\n private dbManager: DatabaseManager;\n private monitorInterval?: NodeJS.Timeout;\n private lastActivityTime: Date;\n private isMonitoring: boolean = false;\n private projectRoot: string;\n\n // Track context state to avoid duplicate saves\n private lastContextCheck: {\n tokens: number;\n percentage: number;\n savedAt?: Date;\n } = { tokens: 0, percentage: 0 };\n\n constructor(\n frameManager: FrameManager,\n dbManager: DatabaseManager,\n projectRoot: string,\n config?: Partial<MonitorConfig>\n ) {\n super();\n\n this.frameManager = frameManager;\n this.dbManager = dbManager;\n this.projectRoot = projectRoot;\n this.lastActivityTime = new Date();\n\n // Initialize config with defaults\n this.config = {\n contextWarningThreshold: 0.6,\n contextCriticalThreshold: 0.7,\n contextAutoSaveThreshold: 0.85,\n idleTimeoutMinutes: 5,\n sessionEndHandoff: true,\n checkIntervalSeconds: 30,\n autoSaveLedger: true,\n autoGenerateHandoff: true,\n autoCompactOnThreshold: false,\n ...config,\n };\n\n // Initialize components\n this.clearSurvival = new ClearSurvival(\n frameManager,\n dbManager,\n new HandoffGenerator(frameManager, dbManager, projectRoot),\n projectRoot\n );\n\n this.handoffGenerator = new HandoffGenerator(\n frameManager,\n dbManager,\n projectRoot\n );\n }\n\n /**\n * Start monitoring session\n */\n async start(): Promise<void> {\n if (this.isMonitoring) return;\n\n this.isMonitoring = true;\n this.lastActivityTime = new Date();\n\n // Start monitoring loop\n this.monitorInterval = setInterval(\n () => this.checkSession(),\n this.config.checkIntervalSeconds * 1000\n );\n\n // Register exit handlers\n this.registerExitHandlers();\n\n this.emit('monitor:started');\n console.log('\uD83D\uDD0D Session monitor started');\n }\n\n /**\n * Stop monitoring\n */\n async stop(): Promise<void> {\n if (!this.isMonitoring) return;\n\n this.isMonitoring = false;\n\n if (this.monitorInterval) {\n clearInterval(this.monitorInterval);\n this.monitorInterval = undefined;\n }\n\n // Generate final handoff if configured\n if (this.config.sessionEndHandoff) {\n await this.generateHandoff('session_end');\n }\n\n this.emit('monitor:stopped');\n console.log('\uD83D\uDED1 Session monitor stopped');\n }\n\n /**\n * Main monitoring check\n */\n private async checkSession(): Promise<void> {\n try {\n // Check context usage\n await this.checkContextUsage();\n\n // Check for idle timeout\n await this.checkIdleTimeout();\n\n // Check for other triggers\n await this.checkCustomTriggers();\n } catch (error: unknown) {\n console.error('Monitor check failed:', error);\n this.emit('monitor:error', error);\n }\n }\n\n /**\n * Check context usage and trigger actions\n */\n private async checkContextUsage(): Promise<void> {\n const currentTokens = await this.estimateTokens();\n const maxTokens = 100000; // Claude's approximate limit\n const percentage = currentTokens / maxTokens;\n\n // Store for comparison\n this.lastContextCheck = {\n tokens: currentTokens,\n percentage,\n savedAt: this.lastContextCheck.savedAt,\n };\n\n // Emit usage update\n this.emit('context:usage', {\n tokens: currentTokens,\n maxTokens,\n percentage,\n status: this.getContextStatus(percentage),\n });\n\n // Check thresholds and take action\n if (percentage >= this.config.contextAutoSaveThreshold) {\n await this.handleCriticalContext();\n } else if (percentage >= this.config.contextCriticalThreshold) {\n await this.handleHighContext();\n } else if (percentage >= this.config.contextWarningThreshold) {\n this.handleWarningContext();\n }\n }\n\n /**\n * Handle critical context (>85%)\n */\n private async handleCriticalContext(): Promise<void> {\n // Avoid duplicate saves within 5 minutes\n if (this.lastContextCheck.savedAt) {\n const minsSinceLastSave =\n (Date.now() - this.lastContextCheck.savedAt.getTime()) / 60000;\n if (minsSinceLastSave < 5) return;\n }\n\n console.log('\uD83D\uDD34 Critical context usage - auto-saving ledger');\n\n if (this.config.autoSaveLedger) {\n const ledger = await this.clearSurvival.saveContinuityLedger();\n this.lastContextCheck.savedAt = new Date();\n\n this.emit('context:ledger_saved', {\n compression: ledger.compression_ratio,\n frames: ledger.active_frame_stack.length,\n tasks: ledger.active_tasks.length,\n });\n\n // Suggest clear\n console.log('\uD83D\uDCA1 Ledger saved. Run /clear to reset context');\n\n if (this.config.autoCompactOnThreshold) {\n // This would trigger actual compaction\n // For now, just emit event for external handling\n this.emit('context:suggest_clear');\n }\n }\n }\n\n /**\n * Handle high context (70-85%)\n */\n private async handleHighContext(): Promise<void> {\n console.log('\u26A0\uFE0F High context usage - consider saving ledger');\n\n this.emit('context:high', {\n percentage: this.lastContextCheck.percentage,\n suggestion: 'Run: stackmemory clear --save',\n });\n }\n\n /**\n * Handle warning context (60-70%)\n */\n private handleWarningContext(): void {\n this.emit('context:warning', {\n percentage: this.lastContextCheck.percentage,\n });\n }\n\n /**\n * Check for idle timeout\n */\n private async checkIdleTimeout(): Promise<void> {\n const idleMinutes = (Date.now() - this.lastActivityTime.getTime()) / 60000;\n\n if (idleMinutes >= this.config.idleTimeoutMinutes) {\n if (this.config.autoGenerateHandoff) {\n await this.generateHandoff('idle_timeout');\n }\n }\n }\n\n /**\n * Generate handoff document\n */\n private async generateHandoff(trigger: string): Promise<void> {\n try {\n const sessionId = await this.dbManager.getCurrentSessionId();\n const handoff = await this.handoffGenerator.generateHandoff(sessionId);\n\n this.emit('handoff:generated', {\n trigger,\n sessionDuration: handoff.session_duration_minutes,\n tasksActive: handoff.active_tasks.filter(\n (t) => t.status !== 'completed'\n ).length,\n });\n\n console.log(`\uD83D\uDCCB Handoff generated (trigger: ${trigger})`);\n } catch (error: unknown) {\n console.error('Failed to generate handoff:', error);\n }\n }\n\n /**\n * Register process exit handlers\n */\n private registerExitHandlers(): void {\n const exitHandler = async (signal: string) => {\n console.log(`\\n\uD83D\uDCE6 Received ${signal}, saving session state...`);\n\n try {\n // Save ledger if context is significant\n if (this.lastContextCheck.percentage > 0.3) {\n await this.clearSurvival.saveContinuityLedger();\n console.log('\u2705 Continuity ledger saved');\n }\n\n // Generate handoff\n if (this.config.sessionEndHandoff) {\n await this.generateHandoff('process_exit');\n console.log('\u2705 Handoff document generated');\n }\n } catch (error: unknown) {\n console.error('Error during exit handling:', error);\n } finally {\n process.exit(0);\n }\n };\n\n // Handle various exit signals\n process.once('SIGINT', () => exitHandler('SIGINT'));\n process.once('SIGTERM', () => exitHandler('SIGTERM'));\n process.once('beforeExit', () => exitHandler('beforeExit'));\n }\n\n /**\n * Update last activity time\n */\n updateActivity(): void {\n this.lastActivityTime = new Date();\n }\n\n /**\n * Check custom triggers (extensible)\n */\n private async checkCustomTriggers(): Promise<void> {\n // Load custom triggers from hooks directory\n const hooksDir = path.join(this.projectRoot, '.stackmemory', 'hooks');\n\n try {\n await fs.access(hooksDir);\n const hooks = await fs.readdir(hooksDir);\n\n for (const hook of hooks) {\n if (hook.startsWith('monitor_') && hook.endsWith('.js')) {\n try {\n const hookPath = path.join(hooksDir, hook);\n const hookModule = await import(hookPath);\n\n if (hookModule.check) {\n const shouldTrigger = await hookModule.check({\n contextPercentage: this.lastContextCheck.percentage,\n idleMinutes:\n (Date.now() - this.lastActivityTime.getTime()) / 60000,\n frameCount: (await this.frameManager.getStack()).frames.length,\n });\n\n if (shouldTrigger && hookModule.action) {\n await hookModule.action({\n clearSurvival: this.clearSurvival,\n handoffGenerator: this.handoffGenerator,\n frameManager: this.frameManager,\n });\n }\n }\n } catch (error: unknown) {\n console.error(`Hook ${hook} failed:`, error);\n }\n }\n }\n } catch {\n // No hooks directory\n }\n }\n\n /**\n * Estimate current token usage\n */\n private async estimateTokens(): Promise<number> {\n const sessionId = await this.dbManager.getCurrentSessionId();\n const frames = await this.dbManager.getRecentFrames(sessionId, 100);\n const traces = await this.dbManager.getRecentTraces(sessionId, 100);\n\n // Rough estimation\n return frames.length * 200 + traces.length * 100;\n }\n\n /**\n * Get context status based on percentage\n */\n private getContextStatus(percentage: number): string {\n if (percentage >= this.config.contextAutoSaveThreshold) return 'critical';\n if (percentage >= this.config.contextCriticalThreshold) return 'high';\n if (percentage >= this.config.contextWarningThreshold) return 'warning';\n return 'ok';\n }\n\n /**\n * Get current monitor status\n */\n getStatus(): {\n isMonitoring: boolean;\n lastActivity: Date;\n contextUsage: typeof this.lastContextCheck;\n config: MonitorConfig;\n } {\n return {\n isMonitoring: this.isMonitoring,\n lastActivity: this.lastActivityTime,\n contextUsage: this.lastContextCheck,\n config: this.config,\n };\n }\n\n /**\n * Update configuration\n */\n updateConfig(config: Partial<MonitorConfig>): void {\n this.config = { ...this.config, ...config };\n this.emit('config:updated', this.config);\n }\n}\n"],
4
+ "sourcesContent": ["/**\n * Session Monitor for StackMemory\n * Automatically triggers features based on session events\n */\n\nimport { EventEmitter } from 'events';\nimport { ClearSurvival } from '../session/clear-survival.js';\nimport { HandoffGenerator } from '../session/handoff-generator.js';\nimport { FrameManager } from '../context/index.js';\nimport { DatabaseManager } from '../storage/database-manager.js';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\n\nexport interface MonitorConfig {\n // Context thresholds\n contextWarningThreshold: number; // Default 0.6 (60%)\n contextCriticalThreshold: number; // Default 0.7 (70%)\n contextAutoSaveThreshold: number; // Default 0.85 (85%)\n\n // Handoff triggers\n idleTimeoutMinutes: number; // Default 5 minutes\n sessionEndHandoff: boolean; // Default true\n\n // Monitoring intervals\n checkIntervalSeconds: number; // Default 30 seconds\n\n // Auto actions\n autoSaveLedger: boolean; // Default true\n autoGenerateHandoff: boolean; // Default true\n autoCompactOnThreshold: boolean; // Default false\n}\n\nexport class SessionMonitor extends EventEmitter {\n private config: MonitorConfig;\n private clearSurvival: ClearSurvival;\n private handoffGenerator: HandoffGenerator;\n private frameManager: FrameManager;\n private dbManager: DatabaseManager;\n private monitorInterval?: NodeJS.Timeout;\n private lastActivityTime: Date;\n private isMonitoring: boolean = false;\n private projectRoot: string;\n\n // Track context state to avoid duplicate saves\n private lastContextCheck: {\n tokens: number;\n percentage: number;\n savedAt?: Date;\n } = { tokens: 0, percentage: 0 };\n\n constructor(\n frameManager: FrameManager,\n dbManager: DatabaseManager,\n projectRoot: string,\n config?: Partial<MonitorConfig>\n ) {\n super();\n\n this.frameManager = frameManager;\n this.dbManager = dbManager;\n this.projectRoot = projectRoot;\n this.lastActivityTime = new Date();\n\n // Initialize config with defaults\n this.config = {\n contextWarningThreshold: 0.6,\n contextCriticalThreshold: 0.7,\n contextAutoSaveThreshold: 0.85,\n idleTimeoutMinutes: 5,\n sessionEndHandoff: true,\n checkIntervalSeconds: 30,\n autoSaveLedger: true,\n autoGenerateHandoff: true,\n autoCompactOnThreshold: false,\n ...config,\n };\n\n // Initialize components\n this.clearSurvival = new ClearSurvival(\n frameManager,\n dbManager,\n new HandoffGenerator(frameManager, dbManager, projectRoot),\n projectRoot\n );\n\n this.handoffGenerator = new HandoffGenerator(\n frameManager,\n dbManager,\n projectRoot\n );\n }\n\n /**\n * Start monitoring session\n */\n async start(): Promise<void> {\n if (this.isMonitoring) return;\n\n this.isMonitoring = true;\n this.lastActivityTime = new Date();\n\n // Start monitoring loop\n this.monitorInterval = setInterval(\n () => this.checkSession(),\n this.config.checkIntervalSeconds * 1000\n );\n\n // Register exit handlers\n this.registerExitHandlers();\n\n this.emit('monitor:started');\n console.log('\uD83D\uDD0D Session monitor started');\n }\n\n /**\n * Stop monitoring\n */\n async stop(): Promise<void> {\n if (!this.isMonitoring) return;\n\n this.isMonitoring = false;\n\n if (this.monitorInterval) {\n clearInterval(this.monitorInterval);\n this.monitorInterval = undefined;\n }\n\n // Generate final handoff if configured\n if (this.config.sessionEndHandoff) {\n await this.generateHandoff('session_end');\n }\n\n this.emit('monitor:stopped');\n console.log('\uD83D\uDED1 Session monitor stopped');\n }\n\n /**\n * Main monitoring check\n */\n private async checkSession(): Promise<void> {\n try {\n // Check context usage\n await this.checkContextUsage();\n\n // Check for idle timeout\n await this.checkIdleTimeout();\n\n // Check for other triggers\n await this.checkCustomTriggers();\n } catch (error: unknown) {\n console.error('Monitor check failed:', error);\n this.emit('monitor:error', error);\n }\n }\n\n /**\n * Check context usage and trigger actions\n */\n private async checkContextUsage(): Promise<void> {\n const currentTokens = await this.estimateTokens();\n const maxTokens = 100000; // Claude's approximate limit\n const percentage = currentTokens / maxTokens;\n\n // Store for comparison\n this.lastContextCheck = {\n tokens: currentTokens,\n percentage,\n savedAt: this.lastContextCheck.savedAt,\n };\n\n // Emit usage update\n this.emit('context:usage', {\n tokens: currentTokens,\n maxTokens,\n percentage,\n status: this.getContextStatus(percentage),\n });\n\n // Check thresholds and take action\n if (percentage >= this.config.contextAutoSaveThreshold) {\n await this.handleCriticalContext();\n } else if (percentage >= this.config.contextCriticalThreshold) {\n await this.handleHighContext();\n } else if (percentage >= this.config.contextWarningThreshold) {\n this.handleWarningContext();\n }\n }\n\n /**\n * Handle critical context (>85%)\n */\n private async handleCriticalContext(): Promise<void> {\n // Avoid duplicate saves within 5 minutes\n if (this.lastContextCheck.savedAt) {\n const minsSinceLastSave =\n (Date.now() - this.lastContextCheck.savedAt.getTime()) / 60000;\n if (minsSinceLastSave < 5) return;\n }\n\n console.log('\uD83D\uDD34 Critical context usage - auto-saving ledger');\n\n if (this.config.autoSaveLedger) {\n const ledger = await this.clearSurvival.saveContinuityLedger();\n this.lastContextCheck.savedAt = new Date();\n\n this.emit('context:ledger_saved', {\n compression: ledger.compression_ratio,\n frames: ledger.active_frame_stack.length,\n tasks: ledger.active_tasks.length,\n });\n\n // Suggest clear\n console.log('\uD83D\uDCA1 Ledger saved. Run /clear to reset context');\n\n if (this.config.autoCompactOnThreshold) {\n // This would trigger actual compaction\n // For now, just emit event for external handling\n this.emit('context:suggest_clear');\n }\n }\n }\n\n /**\n * Handle high context (70-85%)\n */\n private async handleHighContext(): Promise<void> {\n console.log('\u26A0\uFE0F High context usage - consider saving ledger');\n\n this.emit('context:high', {\n percentage: this.lastContextCheck.percentage,\n suggestion: 'Run: stackmemory clear --save',\n });\n }\n\n /**\n * Handle warning context (60-70%)\n */\n private handleWarningContext(): void {\n this.emit('context:warning', {\n percentage: this.lastContextCheck.percentage,\n });\n }\n\n /**\n * Check for idle timeout\n */\n private async checkIdleTimeout(): Promise<void> {\n const idleMinutes = (Date.now() - this.lastActivityTime.getTime()) / 60000;\n\n if (idleMinutes >= this.config.idleTimeoutMinutes) {\n if (this.config.autoGenerateHandoff) {\n await this.generateHandoff('idle_timeout');\n }\n }\n }\n\n /**\n * Generate handoff document\n */\n private async generateHandoff(trigger: string): Promise<void> {\n try {\n const sessionId = await this.dbManager.getCurrentSessionId();\n const handoff = await this.handoffGenerator.generateHandoff(sessionId);\n\n this.emit('handoff:generated', {\n trigger,\n sessionDuration: handoff.session_duration_minutes,\n tasksActive: handoff.active_tasks.filter(\n (t) => t.status !== 'completed'\n ).length,\n });\n\n console.log(`\uD83D\uDCCB Handoff generated (trigger: ${trigger})`);\n } catch (error: unknown) {\n console.error('Failed to generate handoff:', error);\n }\n }\n\n /**\n * Register process exit handlers\n */\n private registerExitHandlers(): void {\n const exitHandler = async (signal: string) => {\n console.log(`\\n\uD83D\uDCE6 Received ${signal}, saving session state...`);\n\n try {\n // Save ledger if context is significant\n if (this.lastContextCheck.percentage > 0.3) {\n await this.clearSurvival.saveContinuityLedger();\n console.log('\u2705 Continuity ledger saved');\n }\n\n // Generate handoff\n if (this.config.sessionEndHandoff) {\n await this.generateHandoff('process_exit');\n console.log('\u2705 Handoff document generated');\n }\n } catch (error: unknown) {\n console.error('Error during exit handling:', error);\n } finally {\n process.exit(0);\n }\n };\n\n // Handle various exit signals\n process.once('SIGINT', () => exitHandler('SIGINT'));\n process.once('SIGTERM', () => exitHandler('SIGTERM'));\n process.once('beforeExit', () => exitHandler('beforeExit'));\n }\n\n /**\n * Update last activity time\n */\n updateActivity(): void {\n this.lastActivityTime = new Date();\n }\n\n /**\n * Check custom triggers (extensible)\n */\n private async checkCustomTriggers(): Promise<void> {\n // Load custom triggers from hooks directory\n const hooksDir = path.join(this.projectRoot, '.stackmemory', 'hooks');\n\n try {\n await fs.access(hooksDir);\n const hooks = await fs.readdir(hooksDir);\n\n for (const hook of hooks) {\n if (hook.startsWith('monitor_') && hook.endsWith('.js')) {\n try {\n const hookPath = path.join(hooksDir, hook);\n const hookModule = await import(hookPath);\n\n if (hookModule.check) {\n const shouldTrigger = await hookModule.check({\n contextPercentage: this.lastContextCheck.percentage,\n idleMinutes:\n (Date.now() - this.lastActivityTime.getTime()) / 60000,\n frameCount: (await this.frameManager.getStack()).frames.length,\n });\n\n if (shouldTrigger && hookModule.action) {\n await hookModule.action({\n clearSurvival: this.clearSurvival,\n handoffGenerator: this.handoffGenerator,\n frameManager: this.frameManager,\n });\n }\n }\n } catch (error: unknown) {\n console.error(`Hook ${hook} failed:`, error);\n }\n }\n }\n } catch {\n // No hooks directory\n }\n }\n\n /**\n * Estimate current token usage\n */\n private async estimateTokens(): Promise<number> {\n const sessionId = await this.dbManager.getCurrentSessionId();\n const frames = await this.dbManager.getRecentFrames(sessionId, 100);\n const traces = await this.dbManager.getRecentTraces(sessionId, 100);\n\n // Rough estimation\n return frames.length * 200 + traces.length * 100;\n }\n\n /**\n * Get context status based on percentage\n */\n private getContextStatus(percentage: number): string {\n if (percentage >= this.config.contextAutoSaveThreshold) return 'critical';\n if (percentage >= this.config.contextCriticalThreshold) return 'high';\n if (percentage >= this.config.contextWarningThreshold) return 'warning';\n return 'ok';\n }\n\n /**\n * Get current monitor status\n */\n getStatus(): {\n isMonitoring: boolean;\n lastActivity: Date;\n contextUsage: typeof this.lastContextCheck;\n config: MonitorConfig;\n } {\n return {\n isMonitoring: this.isMonitoring,\n lastActivity: this.lastActivityTime,\n contextUsage: this.lastContextCheck,\n config: this.config,\n };\n }\n\n /**\n * Update configuration\n */\n updateConfig(config: Partial<MonitorConfig>): void {\n this.config = { ...this.config, ...config };\n this.emit('config:updated', this.config);\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/performance/lazy-context-loader.ts"],
4
- "sourcesContent": ["/**\n * Lazy Context Loader\n * Deferred loading and progressive enhancement for context data\n */\n\nimport Database from 'better-sqlite3';\nimport { Frame, Anchor, Event } from '../context/frame-manager.js';\nimport { logger } from '../monitoring/logger.js';\n\nexport interface LazyLoadOptions {\n preloadDepth?: number; // How many levels to preload\n chunkSize?: number; // Items per chunk\n priority?: 'recency' | 'relevance' | 'frequency';\n}\n\nexport interface ContextChunk {\n frames: Frame[];\n anchors: Anchor[];\n events: Event[];\n metadata: {\n chunkId: number;\n totalChunks: number;\n hasMore: boolean;\n nextCursor?: string;\n };\n}\n\n/**\n * Lazy proxy for deferred data loading\n */\nexport class LazyProxy<T> {\n private _value?: T;\n private _promise?: Promise<T>;\n private _loader: () => Promise<T>;\n private _loaded = false;\n\n constructor(loader: () => Promise<T>) {\n this._loader = loader;\n }\n\n async get(): Promise<T> {\n if (this._loaded && this._value !== undefined) {\n return this._value;\n }\n\n if (!this._promise) {\n this._promise = this._loader().then((value) => {\n this._value = value;\n this._loaded = true;\n return value;\n });\n }\n\n return this._promise;\n }\n\n isLoaded(): boolean {\n return this._loaded;\n }\n\n peek(): T | undefined {\n return this._value;\n }\n\n reset(): void {\n this._value = undefined;\n this._promise = undefined;\n this._loaded = false;\n }\n}\n\nexport class LazyContextLoader {\n private db: Database.Database;\n private projectId: string;\n\n // Lazy loading registries\n private frameLoaders = new Map<string, LazyProxy<Frame>>();\n private anchorLoaders = new Map<string, LazyProxy<Anchor[]>>();\n private eventLoaders = new Map<string, LazyProxy<Event[]>>();\n\n constructor(db: Database.Database, projectId: string) {\n this.db = db;\n this.projectId = projectId;\n }\n\n /**\n * Create a lazy frame reference\n */\n lazyFrame(frameId: string): LazyProxy<Frame> {\n if (!this.frameLoaders.has(frameId)) {\n this.frameLoaders.set(\n frameId,\n new LazyProxy(async () => {\n const frame = this.loadFrame(frameId);\n if (!frame) {\n throw new Error(`Frame not found: ${frameId}`);\n }\n return frame;\n })\n );\n }\n return this.frameLoaders.get(frameId)!;\n }\n\n /**\n * Create lazy anchor references\n */\n lazyAnchors(frameId: string): LazyProxy<Anchor[]> {\n if (!this.anchorLoaders.has(frameId)) {\n this.anchorLoaders.set(\n frameId,\n new LazyProxy(async () => {\n return this.loadAnchors(frameId);\n })\n );\n }\n return this.anchorLoaders.get(frameId)!;\n }\n\n /**\n * Create lazy event references\n */\n lazyEvents(frameId: string, limit = 100): LazyProxy<Event[]> {\n const key = `${frameId}:${limit}`;\n if (!this.eventLoaders.has(key)) {\n this.eventLoaders.set(\n key,\n new LazyProxy(async () => {\n return this.loadEvents(frameId, limit);\n })\n );\n }\n return this.eventLoaders.get(key)!;\n }\n\n /**\n * Progressive context loading with chunking\n */\n async *loadContextProgressive(\n frameIds: string[],\n options: LazyLoadOptions = {}\n ): AsyncGenerator<ContextChunk, void, unknown> {\n const { chunkSize = 10, priority = 'recency' } = options;\n\n // Sort frame IDs by priority\n const sortedIds = this.sortByPriority(frameIds, priority);\n const totalChunks = Math.ceil(sortedIds.length / chunkSize);\n\n for (let i = 0; i < sortedIds.length; i += chunkSize) {\n const chunkIds = sortedIds.slice(i, i + chunkSize);\n const chunkNumber = Math.floor(i / chunkSize) + 1;\n\n const frames: Frame[] = [];\n const anchors: Anchor[] = [];\n const events: Event[] = [];\n\n // Load chunk data\n for (const frameId of chunkIds) {\n const frame = await this.lazyFrame(frameId).get();\n frames.push(frame);\n\n // Load associated data\n const frameAnchors = await this.lazyAnchors(frameId).get();\n anchors.push(...frameAnchors);\n\n const frameEvents = await this.lazyEvents(frameId).get();\n events.push(...frameEvents);\n }\n\n yield {\n frames,\n anchors,\n events,\n metadata: {\n chunkId: chunkNumber,\n totalChunks,\n hasMore: i + chunkSize < sortedIds.length,\n nextCursor:\n i + chunkSize < sortedIds.length\n ? sortedIds[i + chunkSize]\n : undefined,\n },\n };\n }\n }\n\n /**\n * Preload context data for better performance\n */\n async preloadContext(\n frameIds: string[],\n options: { parallel?: boolean; depth?: number } = {}\n ): Promise<void> {\n const { parallel = true, depth = 1 } = options;\n const startTime = Date.now();\n\n if (parallel) {\n const promises: Promise<any>[] = [];\n\n for (const frameId of frameIds) {\n promises.push(this.lazyFrame(frameId).get());\n\n if (depth > 0) {\n promises.push(this.lazyAnchors(frameId).get());\n }\n\n if (depth > 1) {\n promises.push(this.lazyEvents(frameId).get());\n }\n }\n\n await Promise.all(promises);\n } else {\n for (const frameId of frameIds) {\n await this.lazyFrame(frameId).get();\n\n if (depth > 0) {\n await this.lazyAnchors(frameId).get();\n }\n\n if (depth > 1) {\n await this.lazyEvents(frameId).get();\n }\n }\n }\n\n logger.debug('Context preload complete', {\n frames: frameIds.length,\n depth,\n duration: Date.now() - startTime,\n });\n }\n\n /**\n * Load only frame headers (lightweight)\n */\n async loadFrameHeaders(frameIds: string[]): Promise<Map<string, any>> {\n const placeholders = frameIds.map(() => '?').join(',');\n const query = `\n SELECT id, type, name, state, score, created_at, updated_at\n FROM frames \n WHERE id IN (${placeholders})\n `;\n\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n const headers = new Map<string, any>();\n\n for (const row of rows) {\n headers.set(row.id, {\n id: row.id,\n type: row.type,\n name: row.name,\n state: row.state,\n score: row.score,\n createdAt: row.created_at,\n updatedAt: row.updated_at,\n });\n }\n\n return headers;\n }\n\n /**\n * Stream context data for memory efficiency\n */\n async *streamContext(\n query: string,\n params: any[] = []\n ): AsyncGenerator<Frame | Anchor | Event, void, unknown> {\n const stmt = this.db.prepare(query);\n const iterator = stmt.iterate(...params);\n\n for (const row of iterator) {\n yield row as any;\n }\n }\n\n /**\n * Clear lazy loading cache\n */\n clearCache(): void {\n this.frameLoaders.clear();\n this.anchorLoaders.clear();\n this.eventLoaders.clear();\n }\n\n /**\n * Get cache statistics\n */\n getCacheStats(): {\n frames: number;\n anchors: number;\n events: number;\n loaded: number;\n } {\n let loaded = 0;\n\n for (const loader of this.frameLoaders.values()) {\n if (loader.isLoaded()) loaded++;\n }\n\n for (const loader of this.anchorLoaders.values()) {\n if (loader.isLoaded()) loaded++;\n }\n\n for (const loader of this.eventLoaders.values()) {\n if (loader.isLoaded()) loaded++;\n }\n\n return {\n frames: this.frameLoaders.size,\n anchors: this.anchorLoaders.size,\n events: this.eventLoaders.size,\n loaded,\n };\n }\n\n // Private methods\n\n private loadFrame(frameId: string): Frame | null {\n try {\n const row = this.db\n .prepare('SELECT * FROM frames WHERE id = ?')\n .get(frameId) as any;\n\n if (!row) return null;\n\n return {\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n };\n } catch (error: unknown) {\n // Return mock frame if table doesn't exist (for benchmarking)\n if (frameId.startsWith('frame-')) {\n return {\n id: frameId,\n type: 'mock',\n name: `Mock ${frameId}`,\n state: 'open',\n score: 0.5,\n created_at: Date.now(),\n updated_at: Date.now(),\n metadata: {},\n } as any;\n }\n return null;\n }\n }\n\n private loadAnchors(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 DESC'\n )\n .all(frameId) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n }));\n } catch {\n return []; // Return empty array if table doesn't exist\n }\n }\n\n private loadEvents(frameId: string, limit: number): Event[] {\n try {\n const rows = this.db\n .prepare(\n 'SELECT * FROM events WHERE frame_id = ? ORDER BY timestamp DESC LIMIT ?'\n )\n .all(frameId, limit) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n data: JSON.parse(row.data || '{}'),\n metadata: JSON.parse(row.metadata || '{}'),\n }));\n } catch {\n return []; // Return empty array if table doesn't exist\n }\n }\n\n private sortByPriority(\n frameIds: string[],\n priority: 'recency' | 'relevance' | 'frequency'\n ): string[] {\n try {\n switch (priority) {\n case 'recency': {\n // Get timestamps and sort\n const query = `\n SELECT id, updated_at FROM frames \n WHERE id IN (${frameIds.map(() => '?').join(',')})\n ORDER BY updated_at DESC\n `;\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n return rows.map((r: any) => r.id);\n }\n\n case 'relevance': {\n // Get scores and sort\n const query = `\n SELECT id, score FROM frames \n WHERE id IN (${frameIds.map(() => '?').join(',')})\n ORDER BY score DESC\n `;\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n return rows.map((r: any) => r.id);\n }\n\n case 'frequency': {\n // Get event counts and sort\n const query = `\n SELECT f.id, COUNT(e.id) as event_count\n FROM frames f\n LEFT JOIN events e ON f.id = e.frame_id\n WHERE f.id IN (${frameIds.map(() => '?').join(',')})\n GROUP BY f.id\n ORDER BY event_count DESC\n `;\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n return rows.map((r: any) => r.id);\n }\n\n default:\n return frameIds;\n }\n } catch {\n // Return original order if tables don't exist\n return frameIds;\n }\n }\n}\n"],
4
+ "sourcesContent": ["/**\n * Lazy Context Loader\n * Deferred loading and progressive enhancement for context data\n */\n\nimport Database from 'better-sqlite3';\nimport { Frame, Anchor, Event } from '../context/index.js';\nimport { logger } from '../monitoring/logger.js';\n\nexport interface LazyLoadOptions {\n preloadDepth?: number; // How many levels to preload\n chunkSize?: number; // Items per chunk\n priority?: 'recency' | 'relevance' | 'frequency';\n}\n\nexport interface ContextChunk {\n frames: Frame[];\n anchors: Anchor[];\n events: Event[];\n metadata: {\n chunkId: number;\n totalChunks: number;\n hasMore: boolean;\n nextCursor?: string;\n };\n}\n\n/**\n * Lazy proxy for deferred data loading\n */\nexport class LazyProxy<T> {\n private _value?: T;\n private _promise?: Promise<T>;\n private _loader: () => Promise<T>;\n private _loaded = false;\n\n constructor(loader: () => Promise<T>) {\n this._loader = loader;\n }\n\n async get(): Promise<T> {\n if (this._loaded && this._value !== undefined) {\n return this._value;\n }\n\n if (!this._promise) {\n this._promise = this._loader().then((value) => {\n this._value = value;\n this._loaded = true;\n return value;\n });\n }\n\n return this._promise;\n }\n\n isLoaded(): boolean {\n return this._loaded;\n }\n\n peek(): T | undefined {\n return this._value;\n }\n\n reset(): void {\n this._value = undefined;\n this._promise = undefined;\n this._loaded = false;\n }\n}\n\nexport class LazyContextLoader {\n private db: Database.Database;\n private projectId: string;\n\n // Lazy loading registries\n private frameLoaders = new Map<string, LazyProxy<Frame>>();\n private anchorLoaders = new Map<string, LazyProxy<Anchor[]>>();\n private eventLoaders = new Map<string, LazyProxy<Event[]>>();\n\n constructor(db: Database.Database, projectId: string) {\n this.db = db;\n this.projectId = projectId;\n }\n\n /**\n * Create a lazy frame reference\n */\n lazyFrame(frameId: string): LazyProxy<Frame> {\n if (!this.frameLoaders.has(frameId)) {\n this.frameLoaders.set(\n frameId,\n new LazyProxy(async () => {\n const frame = this.loadFrame(frameId);\n if (!frame) {\n throw new Error(`Frame not found: ${frameId}`);\n }\n return frame;\n })\n );\n }\n return this.frameLoaders.get(frameId)!;\n }\n\n /**\n * Create lazy anchor references\n */\n lazyAnchors(frameId: string): LazyProxy<Anchor[]> {\n if (!this.anchorLoaders.has(frameId)) {\n this.anchorLoaders.set(\n frameId,\n new LazyProxy(async () => {\n return this.loadAnchors(frameId);\n })\n );\n }\n return this.anchorLoaders.get(frameId)!;\n }\n\n /**\n * Create lazy event references\n */\n lazyEvents(frameId: string, limit = 100): LazyProxy<Event[]> {\n const key = `${frameId}:${limit}`;\n if (!this.eventLoaders.has(key)) {\n this.eventLoaders.set(\n key,\n new LazyProxy(async () => {\n return this.loadEvents(frameId, limit);\n })\n );\n }\n return this.eventLoaders.get(key)!;\n }\n\n /**\n * Progressive context loading with chunking\n */\n async *loadContextProgressive(\n frameIds: string[],\n options: LazyLoadOptions = {}\n ): AsyncGenerator<ContextChunk, void, unknown> {\n const { chunkSize = 10, priority = 'recency' } = options;\n\n // Sort frame IDs by priority\n const sortedIds = this.sortByPriority(frameIds, priority);\n const totalChunks = Math.ceil(sortedIds.length / chunkSize);\n\n for (let i = 0; i < sortedIds.length; i += chunkSize) {\n const chunkIds = sortedIds.slice(i, i + chunkSize);\n const chunkNumber = Math.floor(i / chunkSize) + 1;\n\n const frames: Frame[] = [];\n const anchors: Anchor[] = [];\n const events: Event[] = [];\n\n // Load chunk data\n for (const frameId of chunkIds) {\n const frame = await this.lazyFrame(frameId).get();\n frames.push(frame);\n\n // Load associated data\n const frameAnchors = await this.lazyAnchors(frameId).get();\n anchors.push(...frameAnchors);\n\n const frameEvents = await this.lazyEvents(frameId).get();\n events.push(...frameEvents);\n }\n\n yield {\n frames,\n anchors,\n events,\n metadata: {\n chunkId: chunkNumber,\n totalChunks,\n hasMore: i + chunkSize < sortedIds.length,\n nextCursor:\n i + chunkSize < sortedIds.length\n ? sortedIds[i + chunkSize]\n : undefined,\n },\n };\n }\n }\n\n /**\n * Preload context data for better performance\n */\n async preloadContext(\n frameIds: string[],\n options: { parallel?: boolean; depth?: number } = {}\n ): Promise<void> {\n const { parallel = true, depth = 1 } = options;\n const startTime = Date.now();\n\n if (parallel) {\n const promises: Promise<any>[] = [];\n\n for (const frameId of frameIds) {\n promises.push(this.lazyFrame(frameId).get());\n\n if (depth > 0) {\n promises.push(this.lazyAnchors(frameId).get());\n }\n\n if (depth > 1) {\n promises.push(this.lazyEvents(frameId).get());\n }\n }\n\n await Promise.all(promises);\n } else {\n for (const frameId of frameIds) {\n await this.lazyFrame(frameId).get();\n\n if (depth > 0) {\n await this.lazyAnchors(frameId).get();\n }\n\n if (depth > 1) {\n await this.lazyEvents(frameId).get();\n }\n }\n }\n\n logger.debug('Context preload complete', {\n frames: frameIds.length,\n depth,\n duration: Date.now() - startTime,\n });\n }\n\n /**\n * Load only frame headers (lightweight)\n */\n async loadFrameHeaders(frameIds: string[]): Promise<Map<string, any>> {\n const placeholders = frameIds.map(() => '?').join(',');\n const query = `\n SELECT id, type, name, state, score, created_at, updated_at\n FROM frames \n WHERE id IN (${placeholders})\n `;\n\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n const headers = new Map<string, any>();\n\n for (const row of rows) {\n headers.set(row.id, {\n id: row.id,\n type: row.type,\n name: row.name,\n state: row.state,\n score: row.score,\n createdAt: row.created_at,\n updatedAt: row.updated_at,\n });\n }\n\n return headers;\n }\n\n /**\n * Stream context data for memory efficiency\n */\n async *streamContext(\n query: string,\n params: any[] = []\n ): AsyncGenerator<Frame | Anchor | Event, void, unknown> {\n const stmt = this.db.prepare(query);\n const iterator = stmt.iterate(...params);\n\n for (const row of iterator) {\n yield row as any;\n }\n }\n\n /**\n * Clear lazy loading cache\n */\n clearCache(): void {\n this.frameLoaders.clear();\n this.anchorLoaders.clear();\n this.eventLoaders.clear();\n }\n\n /**\n * Get cache statistics\n */\n getCacheStats(): {\n frames: number;\n anchors: number;\n events: number;\n loaded: number;\n } {\n let loaded = 0;\n\n for (const loader of this.frameLoaders.values()) {\n if (loader.isLoaded()) loaded++;\n }\n\n for (const loader of this.anchorLoaders.values()) {\n if (loader.isLoaded()) loaded++;\n }\n\n for (const loader of this.eventLoaders.values()) {\n if (loader.isLoaded()) loaded++;\n }\n\n return {\n frames: this.frameLoaders.size,\n anchors: this.anchorLoaders.size,\n events: this.eventLoaders.size,\n loaded,\n };\n }\n\n // Private methods\n\n private loadFrame(frameId: string): Frame | null {\n try {\n const row = this.db\n .prepare('SELECT * FROM frames WHERE id = ?')\n .get(frameId) as any;\n\n if (!row) return null;\n\n return {\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n };\n } catch (error: unknown) {\n // Return mock frame if table doesn't exist (for benchmarking)\n if (frameId.startsWith('frame-')) {\n return {\n id: frameId,\n type: 'mock',\n name: `Mock ${frameId}`,\n state: 'open',\n score: 0.5,\n created_at: Date.now(),\n updated_at: Date.now(),\n metadata: {},\n } as any;\n }\n return null;\n }\n }\n\n private loadAnchors(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 DESC'\n )\n .all(frameId) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n }));\n } catch {\n return []; // Return empty array if table doesn't exist\n }\n }\n\n private loadEvents(frameId: string, limit: number): Event[] {\n try {\n const rows = this.db\n .prepare(\n 'SELECT * FROM events WHERE frame_id = ? ORDER BY timestamp DESC LIMIT ?'\n )\n .all(frameId, limit) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n data: JSON.parse(row.data || '{}'),\n metadata: JSON.parse(row.metadata || '{}'),\n }));\n } catch {\n return []; // Return empty array if table doesn't exist\n }\n }\n\n private sortByPriority(\n frameIds: string[],\n priority: 'recency' | 'relevance' | 'frequency'\n ): string[] {\n try {\n switch (priority) {\n case 'recency': {\n // Get timestamps and sort\n const query = `\n SELECT id, updated_at FROM frames \n WHERE id IN (${frameIds.map(() => '?').join(',')})\n ORDER BY updated_at DESC\n `;\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n return rows.map((r: any) => r.id);\n }\n\n case 'relevance': {\n // Get scores and sort\n const query = `\n SELECT id, score FROM frames \n WHERE id IN (${frameIds.map(() => '?').join(',')})\n ORDER BY score DESC\n `;\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n return rows.map((r: any) => r.id);\n }\n\n case 'frequency': {\n // Get event counts and sort\n const query = `\n SELECT f.id, COUNT(e.id) as event_count\n FROM frames f\n LEFT JOIN events e ON f.id = e.frame_id\n WHERE f.id IN (${frameIds.map(() => '?').join(',')})\n GROUP BY f.id\n ORDER BY event_count DESC\n `;\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n return rows.map((r: any) => r.id);\n }\n\n default:\n return frameIds;\n }\n } catch {\n // Return original order if tables don't exist\n return frameIds;\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/performance/optimized-frame-context.ts"],
4
- "sourcesContent": ["/**\n * Optimized Frame Context Assembly\n * High-performance context retrieval with caching and batching\n */\n\nimport Database from 'better-sqlite3';\nimport { getQueryCache, createCacheKey } from '../database/query-cache.js';\nimport { logger } from '../monitoring/logger.js';\nimport {\n Frame,\n FrameContext,\n Anchor,\n Event,\n} from '../context/frame-manager.js';\n\nexport interface ContextAssemblyOptions {\n maxEvents?: number;\n includeClosed?: boolean;\n enableCaching?: boolean;\n batchSize?: number;\n}\n\nexport interface OptimizedFrameContext extends FrameContext {\n performance: {\n assemblyTimeMs: number;\n cacheHits: number;\n dbQueries: number;\n totalRows: number;\n };\n}\n\n/**\n * Optimized context assembly with caching and batching\n */\nexport class OptimizedContextAssembler {\n private db: Database.Database;\n private cache = getQueryCache();\n private preparedStatements = new Map<string, Database.Statement>();\n\n constructor(db: Database.Database) {\n this.db = db;\n this.initializePreparedStatements();\n }\n\n /**\n * Get hot stack context with optimizations\n */\n async getHotStackContext(\n activeStack: string[],\n options: ContextAssemblyOptions = {}\n ): Promise<OptimizedFrameContext[]> {\n const startTime = performance.now();\n const stats = {\n cacheHits: 0,\n dbQueries: 0,\n totalRows: 0,\n };\n\n const {\n maxEvents = 20,\n includeClosed = false,\n enableCaching = true,\n batchSize = 10,\n } = options;\n\n try {\n // Batch process frames for better performance\n const contexts: OptimizedFrameContext[] = [];\n\n for (let i = 0; i < activeStack.length; i += batchSize) {\n const batch = activeStack.slice(i, i + batchSize);\n const batchContexts = await this.processBatch(\n batch,\n maxEvents,\n includeClosed,\n enableCaching,\n stats\n );\n contexts.push(...batchContexts);\n }\n\n const assemblyTimeMs = performance.now() - startTime;\n\n // Add performance stats to each context\n return contexts.map((context: any) => ({\n ...context,\n performance: {\n assemblyTimeMs: assemblyTimeMs / contexts.length,\n ...stats,\n },\n }));\n } catch (error: unknown) {\n logger.error('Failed to assemble hot stack context', error as Error, {\n activeStack,\n options,\n });\n throw error;\n }\n }\n\n /**\n * Get single frame context with full optimization\n */\n async getFrameContext(\n frameId: string,\n options: ContextAssemblyOptions = {}\n ): Promise<OptimizedFrameContext | null> {\n const startTime = performance.now();\n const stats = { cacheHits: 0, dbQueries: 0, totalRows: 0 };\n\n const { maxEvents = 50, enableCaching = true } = options;\n\n // Check cache first\n const cacheKey = createCacheKey('frame_context', [frameId, maxEvents]);\n if (enableCaching) {\n const cached = this.cache.getFrameContext(cacheKey);\n if (cached) {\n stats.cacheHits++;\n return {\n ...cached,\n performance: {\n assemblyTimeMs: performance.now() - startTime,\n ...stats,\n },\n };\n }\n }\n\n try {\n const context = await this.assembleFrameContext(\n frameId,\n maxEvents,\n stats\n );\n\n if (!context) return null;\n\n // Cache the result\n if (enableCaching) {\n this.cache.cacheFrameContext(cacheKey, context);\n }\n\n const result: OptimizedFrameContext = {\n ...context,\n performance: {\n assemblyTimeMs: performance.now() - startTime,\n ...stats,\n },\n };\n\n return result;\n } catch (error: unknown) {\n logger.error('Failed to get frame context', error as Error, { frameId });\n throw error;\n }\n }\n\n /**\n * Process a batch of frames efficiently\n */\n private async processBatch(\n frameIds: string[],\n maxEvents: number,\n includeClosed: boolean,\n enableCaching: boolean,\n stats: { cacheHits: number; dbQueries: number; totalRows: number }\n ): Promise<OptimizedFrameContext[]> {\n const contexts: OptimizedFrameContext[] = [];\n\n // Get cached contexts first\n const uncachedIds = [];\n for (const frameId of frameIds) {\n const cacheKey = createCacheKey('frame_context', [frameId, maxEvents]);\n if (enableCaching) {\n const cached = this.cache.getFrameContext(cacheKey);\n if (cached) {\n stats.cacheHits++;\n contexts.push(cached);\n continue;\n }\n }\n uncachedIds.push(frameId);\n }\n\n if (uncachedIds.length === 0) {\n return contexts;\n }\n\n // Batch fetch uncached frames\n const frames = await this.batchGetFrames(uncachedIds, stats);\n const allEvents = await this.batchGetEvents(uncachedIds, maxEvents, stats);\n const allAnchors = await this.batchGetAnchors(uncachedIds, stats);\n const allArtifacts = await this.batchGetArtifacts(uncachedIds, stats);\n\n // Assemble contexts from batched data\n for (const frameId of uncachedIds) {\n const frame = frames.get(frameId);\n if (!frame || (!includeClosed && frame.state === 'closed')) {\n continue;\n }\n\n const context: FrameContext = {\n frameId,\n header: {\n goal: frame.name,\n constraints: this.extractConstraints(frame.inputs),\n definitions: frame.inputs.definitions,\n },\n anchors: allAnchors.get(frameId) || [],\n recentEvents: allEvents.get(frameId) || [],\n activeArtifacts: allArtifacts.get(frameId) || [],\n };\n\n // Cache the context\n if (enableCaching) {\n const cacheKey = createCacheKey('frame_context', [frameId, maxEvents]);\n this.cache.cacheFrameContext(cacheKey, context);\n }\n\n contexts.push(context as OptimizedFrameContext);\n }\n\n return contexts;\n }\n\n /**\n * Batch get frames with single query\n */\n private async batchGetFrames(\n frameIds: string[],\n stats: { dbQueries: number; totalRows: number }\n ): Promise<Map<string, Frame>> {\n if (frameIds.length === 0) return new Map();\n\n const stmt = this.preparedStatements.get('batch_frames');\n if (!stmt) throw new Error('Prepared statement not found: batch_frames');\n\n const placeholders = frameIds.map(() => '?').join(',');\n const query = `SELECT * FROM frames WHERE frame_id IN (${placeholders})`;\n\n stats.dbQueries++;\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n stats.totalRows += rows.length;\n\n const frameMap = new Map<string, Frame>();\n for (const row of rows) {\n frameMap.set(row.frame_id, {\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n });\n }\n\n return frameMap;\n }\n\n /**\n * Batch get events for multiple frames\n */\n private async batchGetEvents(\n frameIds: string[],\n maxEvents: number,\n stats: { dbQueries: number; totalRows: number }\n ): Promise<Map<string, Event[]>> {\n if (frameIds.length === 0) return new Map();\n\n const placeholders = frameIds.map(() => '?').join(',');\n const query = `\n SELECT *, ROW_NUMBER() OVER (PARTITION BY frame_id ORDER BY seq DESC) as rn\n FROM events \n WHERE frame_id IN (${placeholders}) \n AND rn <= ${maxEvents}\n ORDER BY frame_id, seq DESC\n `;\n\n stats.dbQueries++;\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n stats.totalRows += rows.length;\n\n const eventMap = new Map<string, Event[]>();\n for (const row of rows) {\n if (!eventMap.has(row.frame_id)) {\n eventMap.set(row.frame_id, []);\n }\n eventMap.get(row.frame_id)!.push({\n ...row,\n payload: JSON.parse(row.payload),\n });\n }\n\n return eventMap;\n }\n\n /**\n * Batch get anchors for multiple frames\n */\n private async batchGetAnchors(\n frameIds: string[],\n stats: { dbQueries: number; totalRows: number }\n ): Promise<Map<string, Anchor[]>> {\n if (frameIds.length === 0) return new Map();\n\n const placeholders = frameIds.map(() => '?').join(',');\n const query = `\n SELECT * FROM anchors \n WHERE frame_id IN (${placeholders}) \n ORDER BY frame_id, priority DESC, created_at ASC\n `;\n\n stats.dbQueries++;\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n stats.totalRows += rows.length;\n\n const anchorMap = new Map<string, Anchor[]>();\n for (const row of rows) {\n if (!anchorMap.has(row.frame_id)) {\n anchorMap.set(row.frame_id, []);\n }\n anchorMap.get(row.frame_id)!.push({\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n });\n }\n\n return anchorMap;\n }\n\n /**\n * Batch get active artifacts for multiple frames\n */\n private async batchGetArtifacts(\n frameIds: string[],\n stats: { dbQueries: number; totalRows: number }\n ): Promise<Map<string, string[]>> {\n if (frameIds.length === 0) return new Map();\n\n const placeholders = frameIds.map(() => '?').join(',');\n const query = `\n SELECT frame_id, payload\n FROM events \n WHERE frame_id IN (${placeholders}) \n AND event_type = 'artifact'\n ORDER BY frame_id, ts DESC\n `;\n\n stats.dbQueries++;\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n stats.totalRows += rows.length;\n\n const artifactMap = new Map<string, string[]>();\n for (const row of rows) {\n const payload = JSON.parse(row.payload);\n if (!artifactMap.has(row.frame_id)) {\n artifactMap.set(row.frame_id, []);\n }\n if (payload.path) {\n artifactMap.get(row.frame_id)!.push(payload.path);\n }\n }\n\n return artifactMap;\n }\n\n /**\n * Assemble single frame context\n */\n private async assembleFrameContext(\n frameId: string,\n maxEvents: number,\n stats: { dbQueries: number; totalRows: number }\n ): Promise<FrameContext | null> {\n // Single frame operations - these could be further optimized with prepared statements\n const frame = await this.batchGetFrames([frameId], stats).then((map) =>\n map.get(frameId)\n );\n if (!frame) return null;\n\n const [events, anchors, artifacts] = await Promise.all([\n this.batchGetEvents([frameId], maxEvents, stats).then(\n (map) => map.get(frameId) || []\n ),\n this.batchGetAnchors([frameId], stats).then(\n (map) => map.get(frameId) || []\n ),\n this.batchGetArtifacts([frameId], stats).then(\n (map) => map.get(frameId) || []\n ),\n ]);\n\n return {\n frameId,\n header: {\n goal: frame.name,\n constraints: this.extractConstraints(frame.inputs),\n definitions: frame.inputs.definitions,\n },\n anchors,\n recentEvents: events,\n activeArtifacts: artifacts,\n };\n }\n\n /**\n * Extract constraints from frame inputs\n */\n private extractConstraints(inputs: Record<string, any>): string[] {\n const constraints: string[] = [];\n\n if (inputs.constraints && Array.isArray(inputs.constraints)) {\n constraints.push(...inputs.constraints);\n }\n\n if (inputs.requirements && Array.isArray(inputs.requirements)) {\n constraints.push(...inputs.requirements);\n }\n\n if (inputs.limitations && Array.isArray(inputs.limitations)) {\n constraints.push(...inputs.limitations);\n }\n\n return constraints;\n }\n\n /**\n * Initialize prepared statements for common queries\n */\n private initializePreparedStatements(): void {\n try {\n // Single frame query\n this.preparedStatements.set(\n 'single_frame',\n this.db.prepare('SELECT * FROM frames WHERE frame_id = ?')\n );\n\n // Frame events with limit\n this.preparedStatements.set(\n 'frame_events',\n this.db.prepare(\n 'SELECT * FROM events WHERE frame_id = ? ORDER BY seq DESC LIMIT ?'\n )\n );\n\n // Frame anchors\n this.preparedStatements.set(\n 'frame_anchors',\n this.db.prepare(\n 'SELECT * FROM anchors WHERE frame_id = ? ORDER BY priority DESC, created_at ASC'\n )\n );\n\n logger.info(\n 'Prepared statements initialized for optimized context assembly'\n );\n } catch (error: unknown) {\n logger.error('Failed to initialize prepared statements', error as Error);\n throw error;\n }\n }\n\n /**\n * Clear cache and reset prepared statements\n */\n cleanup(): void {\n this.cache.clear();\n // Modern better-sqlite3 automatically handles cleanup\n this.preparedStatements.clear();\n }\n}\n"],
4
+ "sourcesContent": ["/**\n * Optimized Frame Context Assembly\n * High-performance context retrieval with caching and batching\n */\n\nimport Database from 'better-sqlite3';\nimport { getQueryCache, createCacheKey } from '../database/query-cache.js';\nimport { logger } from '../monitoring/logger.js';\nimport {\n Frame,\n FrameContext,\n Anchor,\n Event,\n} from '../context/index.js';\n\nexport interface ContextAssemblyOptions {\n maxEvents?: number;\n includeClosed?: boolean;\n enableCaching?: boolean;\n batchSize?: number;\n}\n\nexport interface OptimizedFrameContext extends FrameContext {\n performance: {\n assemblyTimeMs: number;\n cacheHits: number;\n dbQueries: number;\n totalRows: number;\n };\n}\n\n/**\n * Optimized context assembly with caching and batching\n */\nexport class OptimizedContextAssembler {\n private db: Database.Database;\n private cache = getQueryCache();\n private preparedStatements = new Map<string, Database.Statement>();\n\n constructor(db: Database.Database) {\n this.db = db;\n this.initializePreparedStatements();\n }\n\n /**\n * Get hot stack context with optimizations\n */\n async getHotStackContext(\n activeStack: string[],\n options: ContextAssemblyOptions = {}\n ): Promise<OptimizedFrameContext[]> {\n const startTime = performance.now();\n const stats = {\n cacheHits: 0,\n dbQueries: 0,\n totalRows: 0,\n };\n\n const {\n maxEvents = 20,\n includeClosed = false,\n enableCaching = true,\n batchSize = 10,\n } = options;\n\n try {\n // Batch process frames for better performance\n const contexts: OptimizedFrameContext[] = [];\n\n for (let i = 0; i < activeStack.length; i += batchSize) {\n const batch = activeStack.slice(i, i + batchSize);\n const batchContexts = await this.processBatch(\n batch,\n maxEvents,\n includeClosed,\n enableCaching,\n stats\n );\n contexts.push(...batchContexts);\n }\n\n const assemblyTimeMs = performance.now() - startTime;\n\n // Add performance stats to each context\n return contexts.map((context: any) => ({\n ...context,\n performance: {\n assemblyTimeMs: assemblyTimeMs / contexts.length,\n ...stats,\n },\n }));\n } catch (error: unknown) {\n logger.error('Failed to assemble hot stack context', error as Error, {\n activeStack,\n options,\n });\n throw error;\n }\n }\n\n /**\n * Get single frame context with full optimization\n */\n async getFrameContext(\n frameId: string,\n options: ContextAssemblyOptions = {}\n ): Promise<OptimizedFrameContext | null> {\n const startTime = performance.now();\n const stats = { cacheHits: 0, dbQueries: 0, totalRows: 0 };\n\n const { maxEvents = 50, enableCaching = true } = options;\n\n // Check cache first\n const cacheKey = createCacheKey('frame_context', [frameId, maxEvents]);\n if (enableCaching) {\n const cached = this.cache.getFrameContext(cacheKey);\n if (cached) {\n stats.cacheHits++;\n return {\n ...cached,\n performance: {\n assemblyTimeMs: performance.now() - startTime,\n ...stats,\n },\n };\n }\n }\n\n try {\n const context = await this.assembleFrameContext(\n frameId,\n maxEvents,\n stats\n );\n\n if (!context) return null;\n\n // Cache the result\n if (enableCaching) {\n this.cache.cacheFrameContext(cacheKey, context);\n }\n\n const result: OptimizedFrameContext = {\n ...context,\n performance: {\n assemblyTimeMs: performance.now() - startTime,\n ...stats,\n },\n };\n\n return result;\n } catch (error: unknown) {\n logger.error('Failed to get frame context', error as Error, { frameId });\n throw error;\n }\n }\n\n /**\n * Process a batch of frames efficiently\n */\n private async processBatch(\n frameIds: string[],\n maxEvents: number,\n includeClosed: boolean,\n enableCaching: boolean,\n stats: { cacheHits: number; dbQueries: number; totalRows: number }\n ): Promise<OptimizedFrameContext[]> {\n const contexts: OptimizedFrameContext[] = [];\n\n // Get cached contexts first\n const uncachedIds = [];\n for (const frameId of frameIds) {\n const cacheKey = createCacheKey('frame_context', [frameId, maxEvents]);\n if (enableCaching) {\n const cached = this.cache.getFrameContext(cacheKey);\n if (cached) {\n stats.cacheHits++;\n contexts.push(cached);\n continue;\n }\n }\n uncachedIds.push(frameId);\n }\n\n if (uncachedIds.length === 0) {\n return contexts;\n }\n\n // Batch fetch uncached frames\n const frames = await this.batchGetFrames(uncachedIds, stats);\n const allEvents = await this.batchGetEvents(uncachedIds, maxEvents, stats);\n const allAnchors = await this.batchGetAnchors(uncachedIds, stats);\n const allArtifacts = await this.batchGetArtifacts(uncachedIds, stats);\n\n // Assemble contexts from batched data\n for (const frameId of uncachedIds) {\n const frame = frames.get(frameId);\n if (!frame || (!includeClosed && frame.state === 'closed')) {\n continue;\n }\n\n const context: FrameContext = {\n frameId,\n header: {\n goal: frame.name,\n constraints: this.extractConstraints(frame.inputs),\n definitions: frame.inputs.definitions,\n },\n anchors: allAnchors.get(frameId) || [],\n recentEvents: allEvents.get(frameId) || [],\n activeArtifacts: allArtifacts.get(frameId) || [],\n };\n\n // Cache the context\n if (enableCaching) {\n const cacheKey = createCacheKey('frame_context', [frameId, maxEvents]);\n this.cache.cacheFrameContext(cacheKey, context);\n }\n\n contexts.push(context as OptimizedFrameContext);\n }\n\n return contexts;\n }\n\n /**\n * Batch get frames with single query\n */\n private async batchGetFrames(\n frameIds: string[],\n stats: { dbQueries: number; totalRows: number }\n ): Promise<Map<string, Frame>> {\n if (frameIds.length === 0) return new Map();\n\n const stmt = this.preparedStatements.get('batch_frames');\n if (!stmt) throw new Error('Prepared statement not found: batch_frames');\n\n const placeholders = frameIds.map(() => '?').join(',');\n const query = `SELECT * FROM frames WHERE frame_id IN (${placeholders})`;\n\n stats.dbQueries++;\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n stats.totalRows += rows.length;\n\n const frameMap = new Map<string, Frame>();\n for (const row of rows) {\n frameMap.set(row.frame_id, {\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n });\n }\n\n return frameMap;\n }\n\n /**\n * Batch get events for multiple frames\n */\n private async batchGetEvents(\n frameIds: string[],\n maxEvents: number,\n stats: { dbQueries: number; totalRows: number }\n ): Promise<Map<string, Event[]>> {\n if (frameIds.length === 0) return new Map();\n\n const placeholders = frameIds.map(() => '?').join(',');\n const query = `\n SELECT *, ROW_NUMBER() OVER (PARTITION BY frame_id ORDER BY seq DESC) as rn\n FROM events \n WHERE frame_id IN (${placeholders}) \n AND rn <= ${maxEvents}\n ORDER BY frame_id, seq DESC\n `;\n\n stats.dbQueries++;\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n stats.totalRows += rows.length;\n\n const eventMap = new Map<string, Event[]>();\n for (const row of rows) {\n if (!eventMap.has(row.frame_id)) {\n eventMap.set(row.frame_id, []);\n }\n eventMap.get(row.frame_id)!.push({\n ...row,\n payload: JSON.parse(row.payload),\n });\n }\n\n return eventMap;\n }\n\n /**\n * Batch get anchors for multiple frames\n */\n private async batchGetAnchors(\n frameIds: string[],\n stats: { dbQueries: number; totalRows: number }\n ): Promise<Map<string, Anchor[]>> {\n if (frameIds.length === 0) return new Map();\n\n const placeholders = frameIds.map(() => '?').join(',');\n const query = `\n SELECT * FROM anchors \n WHERE frame_id IN (${placeholders}) \n ORDER BY frame_id, priority DESC, created_at ASC\n `;\n\n stats.dbQueries++;\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n stats.totalRows += rows.length;\n\n const anchorMap = new Map<string, Anchor[]>();\n for (const row of rows) {\n if (!anchorMap.has(row.frame_id)) {\n anchorMap.set(row.frame_id, []);\n }\n anchorMap.get(row.frame_id)!.push({\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n });\n }\n\n return anchorMap;\n }\n\n /**\n * Batch get active artifacts for multiple frames\n */\n private async batchGetArtifacts(\n frameIds: string[],\n stats: { dbQueries: number; totalRows: number }\n ): Promise<Map<string, string[]>> {\n if (frameIds.length === 0) return new Map();\n\n const placeholders = frameIds.map(() => '?').join(',');\n const query = `\n SELECT frame_id, payload\n FROM events \n WHERE frame_id IN (${placeholders}) \n AND event_type = 'artifact'\n ORDER BY frame_id, ts DESC\n `;\n\n stats.dbQueries++;\n const rows = this.db.prepare(query).all(...frameIds) as any[];\n stats.totalRows += rows.length;\n\n const artifactMap = new Map<string, string[]>();\n for (const row of rows) {\n const payload = JSON.parse(row.payload);\n if (!artifactMap.has(row.frame_id)) {\n artifactMap.set(row.frame_id, []);\n }\n if (payload.path) {\n artifactMap.get(row.frame_id)!.push(payload.path);\n }\n }\n\n return artifactMap;\n }\n\n /**\n * Assemble single frame context\n */\n private async assembleFrameContext(\n frameId: string,\n maxEvents: number,\n stats: { dbQueries: number; totalRows: number }\n ): Promise<FrameContext | null> {\n // Single frame operations - these could be further optimized with prepared statements\n const frame = await this.batchGetFrames([frameId], stats).then((map) =>\n map.get(frameId)\n );\n if (!frame) return null;\n\n const [events, anchors, artifacts] = await Promise.all([\n this.batchGetEvents([frameId], maxEvents, stats).then(\n (map) => map.get(frameId) || []\n ),\n this.batchGetAnchors([frameId], stats).then(\n (map) => map.get(frameId) || []\n ),\n this.batchGetArtifacts([frameId], stats).then(\n (map) => map.get(frameId) || []\n ),\n ]);\n\n return {\n frameId,\n header: {\n goal: frame.name,\n constraints: this.extractConstraints(frame.inputs),\n definitions: frame.inputs.definitions,\n },\n anchors,\n recentEvents: events,\n activeArtifacts: artifacts,\n };\n }\n\n /**\n * Extract constraints from frame inputs\n */\n private extractConstraints(inputs: Record<string, any>): string[] {\n const constraints: string[] = [];\n\n if (inputs.constraints && Array.isArray(inputs.constraints)) {\n constraints.push(...inputs.constraints);\n }\n\n if (inputs.requirements && Array.isArray(inputs.requirements)) {\n constraints.push(...inputs.requirements);\n }\n\n if (inputs.limitations && Array.isArray(inputs.limitations)) {\n constraints.push(...inputs.limitations);\n }\n\n return constraints;\n }\n\n /**\n * Initialize prepared statements for common queries\n */\n private initializePreparedStatements(): void {\n try {\n // Single frame query\n this.preparedStatements.set(\n 'single_frame',\n this.db.prepare('SELECT * FROM frames WHERE frame_id = ?')\n );\n\n // Frame events with limit\n this.preparedStatements.set(\n 'frame_events',\n this.db.prepare(\n 'SELECT * FROM events WHERE frame_id = ? ORDER BY seq DESC LIMIT ?'\n )\n );\n\n // Frame anchors\n this.preparedStatements.set(\n 'frame_anchors',\n this.db.prepare(\n 'SELECT * FROM anchors WHERE frame_id = ? ORDER BY priority DESC, created_at ASC'\n )\n );\n\n logger.info(\n 'Prepared statements initialized for optimized context assembly'\n );\n } catch (error: unknown) {\n logger.error('Failed to initialize prepared statements', error as Error);\n throw error;\n }\n }\n\n /**\n * Clear cache and reset prepared statements\n */\n cleanup(): void {\n this.cache.clear();\n // Modern better-sqlite3 automatically handles cleanup\n this.preparedStatements.clear();\n }\n}\n"],
5
5
  "mappings": "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",
6
6
  "names": []
7
7
  }