@stackmemoryai/stackmemory 0.5.31 → 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.
- package/dist/agents/core/agent-task-manager.js.map +1 -1
- package/dist/cli/claude-sm.js +199 -16
- package/dist/cli/claude-sm.js.map +2 -2
- package/dist/cli/commands/clear.js +1 -1
- package/dist/cli/commands/clear.js.map +1 -1
- package/dist/cli/commands/context.js +1 -12
- package/dist/cli/commands/context.js.map +2 -2
- package/dist/cli/commands/dashboard.js.map +1 -1
- package/dist/cli/commands/discovery.js +1 -1
- package/dist/cli/commands/discovery.js.map +1 -1
- package/dist/cli/commands/handoff.js +1 -1
- package/dist/cli/commands/handoff.js.map +1 -1
- package/dist/cli/commands/linear.js +1 -14
- package/dist/cli/commands/linear.js.map +2 -2
- package/dist/cli/commands/login.js +32 -10
- package/dist/cli/commands/login.js.map +2 -2
- package/dist/cli/commands/migrate.js +80 -22
- package/dist/cli/commands/migrate.js.map +2 -2
- package/dist/cli/commands/model.js +533 -0
- package/dist/cli/commands/model.js.map +7 -0
- package/dist/cli/commands/monitor.js +1 -1
- package/dist/cli/commands/monitor.js.map +1 -1
- package/dist/cli/commands/quality.js +1 -1
- package/dist/cli/commands/quality.js.map +1 -1
- package/dist/cli/commands/ralph.js +93 -28
- package/dist/cli/commands/ralph.js.map +2 -2
- package/dist/cli/commands/service.js +10 -3
- package/dist/cli/commands/service.js.map +2 -2
- package/dist/cli/commands/skills.js +61 -11
- package/dist/cli/commands/skills.js.map +2 -2
- package/dist/cli/commands/sms-notify.js +342 -22
- package/dist/cli/commands/sms-notify.js.map +3 -3
- package/dist/cli/commands/workflow.js +1 -1
- package/dist/cli/commands/workflow.js.map +1 -1
- package/dist/cli/commands/worktree.js +1 -1
- package/dist/cli/commands/worktree.js.map +1 -1
- package/dist/cli/index.js +3 -1
- package/dist/cli/index.js.map +2 -2
- package/dist/core/context/auto-context.js.map +1 -1
- package/dist/core/context/compaction-handler.js.map +2 -2
- package/dist/core/context/context-bridge.js.map +2 -2
- package/dist/core/context/dual-stack-manager.js +24 -8
- package/dist/core/context/dual-stack-manager.js.map +2 -2
- package/dist/core/context/enhanced-rehydration.js.map +1 -1
- package/dist/core/context/frame-database.js +41 -5
- package/dist/core/context/frame-database.js.map +2 -2
- package/dist/core/context/frame-digest.js +6 -1
- package/dist/core/context/frame-digest.js.map +2 -2
- package/dist/core/context/frame-handoff-manager.js.map +1 -1
- package/dist/core/context/frame-lifecycle-hooks.js +119 -0
- package/dist/core/context/frame-lifecycle-hooks.js.map +7 -0
- package/dist/core/context/frame-manager.js +56 -9
- package/dist/core/context/frame-manager.js.map +2 -2
- package/dist/core/context/frame-stack.js +29 -0
- package/dist/core/context/frame-stack.js.map +2 -2
- package/dist/core/context/incremental-gc.js.map +2 -2
- package/dist/core/context/index.js +4 -22
- package/dist/core/context/index.js.map +2 -2
- package/dist/core/context/permission-manager.js +0 -11
- package/dist/core/context/permission-manager.js.map +2 -2
- package/dist/core/context/recursive-context-manager.js +15 -9
- package/dist/core/context/recursive-context-manager.js.map +2 -2
- package/dist/core/context/refactored-frame-manager.js +140 -34
- package/dist/core/context/refactored-frame-manager.js.map +3 -3
- package/dist/core/context/shared-context-layer.js +0 -11
- package/dist/core/context/shared-context-layer.js.map +2 -2
- package/dist/core/context/stack-merge-resolver.js.map +1 -1
- package/dist/core/context/validation.js +6 -1
- package/dist/core/context/validation.js.map +2 -2
- package/dist/core/database/database-adapter.js.map +1 -1
- package/dist/core/database/paradedb-adapter.js.map +1 -1
- package/dist/core/database/query-router.js.map +1 -1
- package/dist/core/database/sqlite-adapter.js.map +1 -1
- package/dist/core/digest/frame-digest-integration.js.map +1 -1
- package/dist/core/digest/hybrid-digest-generator.js.map +1 -1
- package/dist/core/digest/types.js.map +1 -1
- package/dist/core/errors/index.js +249 -0
- package/dist/core/errors/index.js.map +2 -2
- package/dist/core/frame/workflow-templates.js.map +2 -2
- package/dist/core/merge/conflict-detector.js.map +1 -1
- package/dist/core/merge/resolution-engine.js.map +1 -1
- package/dist/core/merge/stack-diff.js.map +1 -1
- package/dist/core/models/fallback-monitor.js +229 -0
- package/dist/core/models/fallback-monitor.js.map +7 -0
- package/dist/core/models/model-router.js +340 -0
- package/dist/core/models/model-router.js.map +7 -0
- package/dist/core/monitoring/error-handler.js +37 -270
- package/dist/core/monitoring/error-handler.js.map +3 -3
- package/dist/core/monitoring/session-monitor.js.map +1 -1
- package/dist/core/performance/lazy-context-loader.js.map +1 -1
- package/dist/core/performance/optimized-frame-context.js.map +1 -1
- package/dist/core/retrieval/context-retriever.js.map +1 -1
- package/dist/core/retrieval/graph-retrieval.js.map +1 -1
- package/dist/core/retrieval/hierarchical-retrieval.js.map +1 -1
- package/dist/core/retrieval/llm-context-retrieval.js.map +1 -1
- package/dist/core/retrieval/retrieval-benchmarks.js.map +1 -1
- package/dist/core/retrieval/summary-generator.js.map +1 -1
- package/dist/core/retrieval/types.js.map +1 -1
- package/dist/core/storage/chromadb-adapter.js.map +1 -1
- package/dist/core/storage/infinite-storage.js.map +1 -1
- package/dist/core/storage/two-tier-storage.js.map +1 -1
- package/dist/features/tasks/task-aware-context.js.map +1 -1
- package/dist/features/web/server/index.js +1 -1
- package/dist/features/web/server/index.js.map +1 -1
- package/dist/hooks/claude-code-whatsapp-hook.js +197 -0
- package/dist/hooks/claude-code-whatsapp-hook.js.map +7 -0
- package/dist/hooks/linear-task-picker.js +1 -1
- package/dist/hooks/linear-task-picker.js.map +2 -2
- package/dist/hooks/schemas.js +105 -1
- package/dist/hooks/schemas.js.map +2 -2
- package/dist/hooks/session-summary.js +5 -1
- package/dist/hooks/session-summary.js.map +2 -2
- package/dist/hooks/sms-action-runner.js +16 -1
- package/dist/hooks/sms-action-runner.js.map +2 -2
- package/dist/hooks/sms-notify.js +4 -2
- package/dist/hooks/sms-notify.js.map +2 -2
- package/dist/hooks/sms-webhook.js +23 -2
- package/dist/hooks/sms-webhook.js.map +2 -2
- package/dist/hooks/whatsapp-commands.js +516 -0
- package/dist/hooks/whatsapp-commands.js.map +7 -0
- package/dist/hooks/whatsapp-scheduler.js +317 -0
- package/dist/hooks/whatsapp-scheduler.js.map +7 -0
- package/dist/hooks/whatsapp-sync.js +409 -0
- package/dist/hooks/whatsapp-sync.js.map +7 -0
- package/dist/index.js +1 -1
- package/dist/index.js.map +1 -1
- package/dist/integrations/mcp/handlers/context-handlers.js.map +1 -1
- package/dist/integrations/mcp/handlers/discovery-handlers.js.map +1 -1
- package/dist/integrations/mcp/server.js +1 -1
- package/dist/integrations/mcp/server.js.map +1 -1
- package/dist/integrations/ralph/bridge/ralph-stackmemory-bridge.js +1 -1
- package/dist/integrations/ralph/bridge/ralph-stackmemory-bridge.js.map +1 -1
- package/dist/integrations/ralph/context/stackmemory-context-loader.js +1 -1
- package/dist/integrations/ralph/context/stackmemory-context-loader.js.map +1 -1
- package/dist/integrations/ralph/learning/pattern-learner.js +1 -1
- package/dist/integrations/ralph/learning/pattern-learner.js.map +1 -1
- package/dist/integrations/ralph/orchestration/multi-loop-orchestrator.js +1 -1
- package/dist/integrations/ralph/orchestration/multi-loop-orchestrator.js.map +1 -1
- package/dist/integrations/ralph/swarm/swarm-coordinator.js +1 -1
- package/dist/integrations/ralph/swarm/swarm-coordinator.js.map +1 -1
- package/dist/integrations/ralph/visualization/ralph-debugger.js +1 -1
- package/dist/integrations/ralph/visualization/ralph-debugger.js.map +1 -1
- package/dist/mcp/stackmemory-mcp-server.js +1 -1
- package/dist/mcp/stackmemory-mcp-server.js.map +1 -1
- package/dist/skills/claude-skills.js.map +1 -1
- package/dist/skills/recursive-agent-orchestrator.js.map +1 -1
- package/dist/skills/unified-rlm-orchestrator.js.map +1 -1
- package/package.json +2 -3
|
@@ -0,0 +1,409 @@
|
|
|
1
|
+
import { fileURLToPath as __fileURLToPath } from 'url';
|
|
2
|
+
import { dirname as __pathDirname } from 'path';
|
|
3
|
+
const __filename = __fileURLToPath(import.meta.url);
|
|
4
|
+
const __dirname = __pathDirname(__filename);
|
|
5
|
+
import { existsSync, readFileSync } from "fs";
|
|
6
|
+
import { join } from "path";
|
|
7
|
+
import { homedir } from "os";
|
|
8
|
+
import {
|
|
9
|
+
sendNotification,
|
|
10
|
+
loadSMSConfig,
|
|
11
|
+
saveSMSConfig
|
|
12
|
+
} from "./sms-notify.js";
|
|
13
|
+
import { writeFileSecure, ensureSecureDir } from "./secure-fs.js";
|
|
14
|
+
import { SyncOptionsSchema, parseConfigSafe } from "./schemas.js";
|
|
15
|
+
import {
|
|
16
|
+
frameLifecycleHooks
|
|
17
|
+
} from "../core/context/frame-lifecycle-hooks.js";
|
|
18
|
+
const SYNC_CONFIG_PATH = join(homedir(), ".stackmemory", "whatsapp-sync.json");
|
|
19
|
+
const DEFAULT_SYNC_OPTIONS = {
|
|
20
|
+
autoSyncOnClose: false,
|
|
21
|
+
minFrameDuration: 30,
|
|
22
|
+
// Skip frames shorter than 30 seconds
|
|
23
|
+
includeDecisions: true,
|
|
24
|
+
includeFiles: true,
|
|
25
|
+
includeTests: true,
|
|
26
|
+
maxDigestLength: 400
|
|
27
|
+
};
|
|
28
|
+
function loadSyncOptions() {
|
|
29
|
+
try {
|
|
30
|
+
if (existsSync(SYNC_CONFIG_PATH)) {
|
|
31
|
+
const data = JSON.parse(readFileSync(SYNC_CONFIG_PATH, "utf8"));
|
|
32
|
+
return parseConfigSafe(
|
|
33
|
+
SyncOptionsSchema,
|
|
34
|
+
{ ...DEFAULT_SYNC_OPTIONS, ...data },
|
|
35
|
+
DEFAULT_SYNC_OPTIONS,
|
|
36
|
+
"whatsapp-sync"
|
|
37
|
+
);
|
|
38
|
+
}
|
|
39
|
+
} catch {
|
|
40
|
+
}
|
|
41
|
+
return { ...DEFAULT_SYNC_OPTIONS };
|
|
42
|
+
}
|
|
43
|
+
function saveSyncOptions(options) {
|
|
44
|
+
try {
|
|
45
|
+
ensureSecureDir(join(homedir(), ".stackmemory"));
|
|
46
|
+
writeFileSecure(SYNC_CONFIG_PATH, JSON.stringify(options, null, 2));
|
|
47
|
+
} catch {
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
function formatDuration(seconds) {
|
|
51
|
+
if (seconds < 60) {
|
|
52
|
+
return `${seconds}s`;
|
|
53
|
+
} else if (seconds < 3600) {
|
|
54
|
+
const mins = Math.floor(seconds / 60);
|
|
55
|
+
const secs = seconds % 60;
|
|
56
|
+
return secs > 0 ? `${mins}m${secs}s` : `${mins}m`;
|
|
57
|
+
} else {
|
|
58
|
+
const hours = Math.floor(seconds / 3600);
|
|
59
|
+
const mins = Math.floor(seconds % 3600 / 60);
|
|
60
|
+
return mins > 0 ? `${hours}h${mins}m` : `${hours}h`;
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
function truncate(text, maxLen) {
|
|
64
|
+
if (text.length <= maxLen) return text;
|
|
65
|
+
return text.slice(0, maxLen - 3) + "...";
|
|
66
|
+
}
|
|
67
|
+
function getStatusSymbol(status) {
|
|
68
|
+
switch (status) {
|
|
69
|
+
case "success":
|
|
70
|
+
return "OK";
|
|
71
|
+
case "failure":
|
|
72
|
+
return "FAIL";
|
|
73
|
+
case "partial":
|
|
74
|
+
return "PARTIAL";
|
|
75
|
+
case "ongoing":
|
|
76
|
+
return "ACTIVE";
|
|
77
|
+
default:
|
|
78
|
+
return "?";
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
function generateMobileDigest(data, options = DEFAULT_SYNC_OPTIONS) {
|
|
82
|
+
const parts = [];
|
|
83
|
+
const maxLen = options.maxDigestLength;
|
|
84
|
+
const header = `FRAME: ${truncate(data.name, 30)} [${data.type}] - ${formatDuration(data.durationSeconds)} ${getStatusSymbol(data.status)}`;
|
|
85
|
+
parts.push(header);
|
|
86
|
+
const activityParts = [];
|
|
87
|
+
if (options.includeFiles && data.filesModified.length > 0) {
|
|
88
|
+
activityParts.push(`FILES: ${data.filesModified.length}`);
|
|
89
|
+
}
|
|
90
|
+
if (data.toolCallCount > 0) {
|
|
91
|
+
activityParts.push(`TOOLS: ${data.toolCallCount}`);
|
|
92
|
+
}
|
|
93
|
+
if (options.includeTests && data.testsRun.length > 0) {
|
|
94
|
+
const passed = data.testsRun.filter((t) => t.status === "passed").length;
|
|
95
|
+
const failed = data.testsRun.filter((t) => t.status === "failed").length;
|
|
96
|
+
if (failed > 0) {
|
|
97
|
+
activityParts.push(`TESTS: ${passed}ok/${failed}fail`);
|
|
98
|
+
} else {
|
|
99
|
+
activityParts.push(`TESTS: ${passed} pass`);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
if (activityParts.length > 0) {
|
|
103
|
+
parts.push(activityParts.join(" | "));
|
|
104
|
+
}
|
|
105
|
+
if (options.includeFiles && data.filesModified.length > 0) {
|
|
106
|
+
const fileList = data.filesModified.slice(0, 3).map((f) => {
|
|
107
|
+
const basename = f.path.split("/").pop() || f.path;
|
|
108
|
+
const op = f.operation.charAt(0).toUpperCase();
|
|
109
|
+
return `${op}:${truncate(basename, 20)}`;
|
|
110
|
+
}).join(", ");
|
|
111
|
+
const more = data.filesModified.length > 3 ? ` +${data.filesModified.length - 3}` : "";
|
|
112
|
+
parts.push(` ${fileList}${more}`);
|
|
113
|
+
}
|
|
114
|
+
if (options.includeDecisions && data.decisions.length > 0) {
|
|
115
|
+
parts.push("");
|
|
116
|
+
parts.push("DECISIONS:");
|
|
117
|
+
data.decisions.slice(0, 3).forEach((d) => {
|
|
118
|
+
parts.push(` ${truncate(d, 60)}`);
|
|
119
|
+
});
|
|
120
|
+
if (data.decisions.length > 3) {
|
|
121
|
+
parts.push(` +${data.decisions.length - 3} more`);
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
if (data.risks.length > 0) {
|
|
125
|
+
parts.push("");
|
|
126
|
+
parts.push("RISKS:");
|
|
127
|
+
data.risks.slice(0, 2).forEach((r) => {
|
|
128
|
+
parts.push(` ${truncate(r, 50)}`);
|
|
129
|
+
});
|
|
130
|
+
}
|
|
131
|
+
const unresolvedErrors = data.errors.filter((e) => !e.resolved);
|
|
132
|
+
if (unresolvedErrors.length > 0) {
|
|
133
|
+
parts.push("");
|
|
134
|
+
parts.push(`ERRORS: ${unresolvedErrors.length} unresolved`);
|
|
135
|
+
unresolvedErrors.slice(0, 2).forEach((e) => {
|
|
136
|
+
parts.push(` ${truncate(e.message, 50)}`);
|
|
137
|
+
});
|
|
138
|
+
}
|
|
139
|
+
parts.push("");
|
|
140
|
+
if (data.status === "success") {
|
|
141
|
+
parts.push("NEXT: commit & test");
|
|
142
|
+
} else if (data.status === "failure") {
|
|
143
|
+
parts.push("NEXT: fix errors");
|
|
144
|
+
} else if (data.status === "partial") {
|
|
145
|
+
parts.push("NEXT: review & continue");
|
|
146
|
+
} else {
|
|
147
|
+
parts.push("NEXT: check status");
|
|
148
|
+
}
|
|
149
|
+
let result = parts.join("\n");
|
|
150
|
+
if (result.length > maxLen) {
|
|
151
|
+
const essentialParts = [header];
|
|
152
|
+
if (activityParts.length > 0) {
|
|
153
|
+
essentialParts.push(activityParts.join(" | "));
|
|
154
|
+
}
|
|
155
|
+
if (options.includeDecisions && data.decisions.length > 0) {
|
|
156
|
+
essentialParts.push("");
|
|
157
|
+
essentialParts.push(`DECISIONS: ${data.decisions.length}`);
|
|
158
|
+
essentialParts.push(` ${truncate(data.decisions[0], 50)}`);
|
|
159
|
+
}
|
|
160
|
+
if (unresolvedErrors.length > 0) {
|
|
161
|
+
essentialParts.push("");
|
|
162
|
+
essentialParts.push(`ERRORS: ${unresolvedErrors.length} unresolved`);
|
|
163
|
+
}
|
|
164
|
+
essentialParts.push("");
|
|
165
|
+
essentialParts.push(
|
|
166
|
+
data.status === "success" ? "NEXT: commit" : "NEXT: review"
|
|
167
|
+
);
|
|
168
|
+
result = essentialParts.join("\n");
|
|
169
|
+
}
|
|
170
|
+
return result.slice(0, maxLen);
|
|
171
|
+
}
|
|
172
|
+
async function getFrameDigestData(frameId) {
|
|
173
|
+
try {
|
|
174
|
+
const digestPath = join(
|
|
175
|
+
homedir(),
|
|
176
|
+
".stackmemory",
|
|
177
|
+
"latest-frame-digest.json"
|
|
178
|
+
);
|
|
179
|
+
if (existsSync(digestPath)) {
|
|
180
|
+
const data = JSON.parse(readFileSync(digestPath, "utf8"));
|
|
181
|
+
if (frameId && data.frameId !== frameId) {
|
|
182
|
+
return null;
|
|
183
|
+
}
|
|
184
|
+
return data;
|
|
185
|
+
}
|
|
186
|
+
return null;
|
|
187
|
+
} catch {
|
|
188
|
+
return null;
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
function storeFrameDigest(data) {
|
|
192
|
+
try {
|
|
193
|
+
ensureSecureDir(join(homedir(), ".stackmemory"));
|
|
194
|
+
const digestPath = join(
|
|
195
|
+
homedir(),
|
|
196
|
+
".stackmemory",
|
|
197
|
+
"latest-frame-digest.json"
|
|
198
|
+
);
|
|
199
|
+
writeFileSecure(digestPath, JSON.stringify(data, null, 2));
|
|
200
|
+
} catch {
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
async function syncContext() {
|
|
204
|
+
const options = loadSyncOptions();
|
|
205
|
+
const data = await getFrameDigestData();
|
|
206
|
+
if (!data) {
|
|
207
|
+
return {
|
|
208
|
+
success: false,
|
|
209
|
+
error: "No context data available. Run a task first."
|
|
210
|
+
};
|
|
211
|
+
}
|
|
212
|
+
return syncFrameData(data, options);
|
|
213
|
+
}
|
|
214
|
+
async function syncFrame(frameId) {
|
|
215
|
+
const options = loadSyncOptions();
|
|
216
|
+
const data = await getFrameDigestData(frameId);
|
|
217
|
+
if (!data) {
|
|
218
|
+
return {
|
|
219
|
+
success: false,
|
|
220
|
+
error: `Frame not found: ${frameId}`
|
|
221
|
+
};
|
|
222
|
+
}
|
|
223
|
+
return syncFrameData(data, options);
|
|
224
|
+
}
|
|
225
|
+
async function syncFrameData(data, options) {
|
|
226
|
+
const config = loadSMSConfig();
|
|
227
|
+
if (!config.enabled) {
|
|
228
|
+
return { success: false, error: "Notifications disabled" };
|
|
229
|
+
}
|
|
230
|
+
if (data.durationSeconds < options.minFrameDuration) {
|
|
231
|
+
return {
|
|
232
|
+
success: false,
|
|
233
|
+
error: `Frame too short (${data.durationSeconds}s < ${options.minFrameDuration}s min)`
|
|
234
|
+
};
|
|
235
|
+
}
|
|
236
|
+
const digest = generateMobileDigest(data, options);
|
|
237
|
+
const payload = {
|
|
238
|
+
type: "custom",
|
|
239
|
+
title: "Context Sync",
|
|
240
|
+
message: digest,
|
|
241
|
+
prompt: {
|
|
242
|
+
type: "options",
|
|
243
|
+
options: [
|
|
244
|
+
{ key: "1", label: "Commit", action: "git add -A && git commit" },
|
|
245
|
+
{ key: "2", label: "Status", action: "stackmemory status" },
|
|
246
|
+
{ key: "3", label: "Continue", action: 'echo "Continuing..."' }
|
|
247
|
+
],
|
|
248
|
+
question: "Action?"
|
|
249
|
+
}
|
|
250
|
+
};
|
|
251
|
+
const result = await sendNotification(payload);
|
|
252
|
+
return {
|
|
253
|
+
success: result.success,
|
|
254
|
+
messageId: result.promptId,
|
|
255
|
+
channel: result.channel,
|
|
256
|
+
error: result.error,
|
|
257
|
+
digestLength: digest.length
|
|
258
|
+
};
|
|
259
|
+
}
|
|
260
|
+
function enableAutoSync(options) {
|
|
261
|
+
const current = loadSyncOptions();
|
|
262
|
+
const updated = {
|
|
263
|
+
...current,
|
|
264
|
+
...options,
|
|
265
|
+
autoSyncOnClose: true
|
|
266
|
+
};
|
|
267
|
+
saveSyncOptions(updated);
|
|
268
|
+
const smsConfig = loadSMSConfig();
|
|
269
|
+
if (!smsConfig.notifyOn.custom) {
|
|
270
|
+
smsConfig.notifyOn.custom = true;
|
|
271
|
+
saveSMSConfig(smsConfig);
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
function disableAutoSync() {
|
|
275
|
+
const current = loadSyncOptions();
|
|
276
|
+
current.autoSyncOnClose = false;
|
|
277
|
+
saveSyncOptions(current);
|
|
278
|
+
}
|
|
279
|
+
function isAutoSyncEnabled() {
|
|
280
|
+
const options = loadSyncOptions();
|
|
281
|
+
return options.autoSyncOnClose;
|
|
282
|
+
}
|
|
283
|
+
async function onFrameClosed(frameData) {
|
|
284
|
+
if (!isAutoSyncEnabled()) {
|
|
285
|
+
return null;
|
|
286
|
+
}
|
|
287
|
+
storeFrameDigest(frameData);
|
|
288
|
+
const options = loadSyncOptions();
|
|
289
|
+
return syncFrameData(frameData, options);
|
|
290
|
+
}
|
|
291
|
+
async function handleFrameCloseHook(data) {
|
|
292
|
+
const digestData = createFrameDigestData(
|
|
293
|
+
data.frame,
|
|
294
|
+
data.events,
|
|
295
|
+
data.anchors
|
|
296
|
+
);
|
|
297
|
+
await onFrameClosed(digestData);
|
|
298
|
+
}
|
|
299
|
+
let hookUnregister = null;
|
|
300
|
+
function registerWhatsAppSyncHook() {
|
|
301
|
+
if (hookUnregister) {
|
|
302
|
+
return hookUnregister;
|
|
303
|
+
}
|
|
304
|
+
hookUnregister = frameLifecycleHooks.onFrameClosed(
|
|
305
|
+
"whatsapp-sync",
|
|
306
|
+
handleFrameCloseHook,
|
|
307
|
+
-10
|
|
308
|
+
// Low priority - run after other hooks
|
|
309
|
+
);
|
|
310
|
+
return () => {
|
|
311
|
+
if (hookUnregister) {
|
|
312
|
+
hookUnregister();
|
|
313
|
+
hookUnregister = null;
|
|
314
|
+
}
|
|
315
|
+
};
|
|
316
|
+
}
|
|
317
|
+
function isHookRegistered() {
|
|
318
|
+
return hookUnregister !== null;
|
|
319
|
+
}
|
|
320
|
+
function createFrameDigestData(frame, events, anchors) {
|
|
321
|
+
const now = Math.floor(Date.now() / 1e3);
|
|
322
|
+
const duration = (frame.closed_at || now) - frame.created_at;
|
|
323
|
+
const filesModified = [];
|
|
324
|
+
const filesSeen = /* @__PURE__ */ new Set();
|
|
325
|
+
events.filter((e) => e.event_type === "tool_call").forEach((e) => {
|
|
326
|
+
const path = e.payload["path"];
|
|
327
|
+
if (path && !filesSeen.has(path)) {
|
|
328
|
+
filesSeen.add(path);
|
|
329
|
+
const toolName = e.payload["tool_name"] || "";
|
|
330
|
+
let operation = "modify";
|
|
331
|
+
if (toolName.includes("Write") || toolName.includes("Create")) {
|
|
332
|
+
operation = "create";
|
|
333
|
+
} else if (toolName.includes("Read")) {
|
|
334
|
+
operation = "read";
|
|
335
|
+
} else if (toolName.includes("Delete")) {
|
|
336
|
+
operation = "delete";
|
|
337
|
+
}
|
|
338
|
+
filesModified.push({ path, operation });
|
|
339
|
+
}
|
|
340
|
+
});
|
|
341
|
+
const testsRun = [];
|
|
342
|
+
events.filter(
|
|
343
|
+
(e) => e.event_type === "tool_result" && String(e.payload["output"] || "").includes("test")
|
|
344
|
+
).forEach((e) => {
|
|
345
|
+
const output = String(e.payload["output"] || "");
|
|
346
|
+
const passMatch = output.match(/(\d+) pass/i);
|
|
347
|
+
const failMatch = output.match(/(\d+) fail/i);
|
|
348
|
+
if (passMatch) {
|
|
349
|
+
testsRun.push({ name: "Tests", status: "passed" });
|
|
350
|
+
}
|
|
351
|
+
if (failMatch && parseInt(failMatch[1]) > 0) {
|
|
352
|
+
testsRun.push({ name: "Tests", status: "failed" });
|
|
353
|
+
}
|
|
354
|
+
});
|
|
355
|
+
const decisions = anchors.filter((a) => a.type === "DECISION").map((a) => a.text);
|
|
356
|
+
const risks = anchors.filter((a) => a.type === "RISK").map((a) => a.text);
|
|
357
|
+
const errors = [];
|
|
358
|
+
events.filter((e) => e.payload["error"] || e.payload["status"] === "error").forEach((e) => {
|
|
359
|
+
const errorMsg = e.payload["error"] || e.payload["message"] || "Unknown error";
|
|
360
|
+
errors.push({
|
|
361
|
+
type: e.payload["type"] || "error",
|
|
362
|
+
message: errorMsg,
|
|
363
|
+
resolved: false
|
|
364
|
+
});
|
|
365
|
+
});
|
|
366
|
+
let status = "ongoing";
|
|
367
|
+
if (frame.closed_at) {
|
|
368
|
+
if (errors.filter((e) => !e.resolved).length > 0) {
|
|
369
|
+
status = "failure";
|
|
370
|
+
} else if (testsRun.some((t) => t.status === "failed") || filesModified.length === 0) {
|
|
371
|
+
status = "partial";
|
|
372
|
+
} else {
|
|
373
|
+
status = "success";
|
|
374
|
+
}
|
|
375
|
+
}
|
|
376
|
+
const toolCallCount = events.filter(
|
|
377
|
+
(e) => e.event_type === "tool_call"
|
|
378
|
+
).length;
|
|
379
|
+
return {
|
|
380
|
+
frameId: frame.frame_id,
|
|
381
|
+
name: frame.name,
|
|
382
|
+
type: frame.type,
|
|
383
|
+
status,
|
|
384
|
+
durationSeconds: duration,
|
|
385
|
+
filesModified: filesModified.filter((f) => f.operation !== "read"),
|
|
386
|
+
testsRun,
|
|
387
|
+
decisions,
|
|
388
|
+
risks,
|
|
389
|
+
toolCallCount,
|
|
390
|
+
errors
|
|
391
|
+
};
|
|
392
|
+
}
|
|
393
|
+
export {
|
|
394
|
+
createFrameDigestData,
|
|
395
|
+
disableAutoSync,
|
|
396
|
+
enableAutoSync,
|
|
397
|
+
generateMobileDigest,
|
|
398
|
+
getFrameDigestData,
|
|
399
|
+
isAutoSyncEnabled,
|
|
400
|
+
isHookRegistered,
|
|
401
|
+
loadSyncOptions,
|
|
402
|
+
onFrameClosed,
|
|
403
|
+
registerWhatsAppSyncHook,
|
|
404
|
+
saveSyncOptions,
|
|
405
|
+
storeFrameDigest,
|
|
406
|
+
syncContext,
|
|
407
|
+
syncFrame
|
|
408
|
+
};
|
|
409
|
+
//# sourceMappingURL=whatsapp-sync.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../src/hooks/whatsapp-sync.ts"],
|
|
4
|
+
"sourcesContent": ["/**\n * WhatsApp Context Sync Engine\n * Push frame digests and context updates to WhatsApp\n *\n * Uses the frame lifecycle hooks system to receive frame close events.\n * Call `registerWhatsAppSyncHook()` to enable automatic sync on frame close.\n */\n\nimport { existsSync, readFileSync } from 'fs';\nimport { join } from 'path';\nimport { homedir } from 'os';\nimport {\n sendNotification,\n loadSMSConfig,\n saveSMSConfig,\n type NotificationPayload,\n} from './sms-notify.js';\nimport { writeFileSecure, ensureSecureDir } from './secure-fs.js';\nimport { SyncOptionsSchema, parseConfigSafe } from './schemas.js';\nimport {\n frameLifecycleHooks,\n type FrameCloseData,\n} from '../core/context/frame-lifecycle-hooks.js';\n\nexport interface SyncOptions {\n autoSyncOnClose: boolean;\n minFrameDuration: number; // seconds, skip short frames\n includeDecisions: boolean;\n includeFiles: boolean;\n includeTests: boolean;\n maxDigestLength: number; // chars, default 400\n}\n\nexport interface SyncResult {\n success: boolean;\n messageId?: string;\n channel?: 'whatsapp' | 'sms';\n error?: string;\n digestLength?: number;\n}\n\nexport interface FrameDigestData {\n frameId: string;\n name: string;\n type: string;\n status: 'success' | 'failure' | 'partial' | 'ongoing';\n durationSeconds: number;\n filesModified: Array<{ path: string; operation: string }>;\n testsRun: Array<{ name: string; status: string }>;\n decisions: string[];\n risks: string[];\n toolCallCount: number;\n errors: Array<{ type: string; message: string; resolved: boolean }>;\n}\n\nconst SYNC_CONFIG_PATH = join(homedir(), '.stackmemory', 'whatsapp-sync.json');\n\nconst DEFAULT_SYNC_OPTIONS: SyncOptions = {\n autoSyncOnClose: false,\n minFrameDuration: 30, // Skip frames shorter than 30 seconds\n includeDecisions: true,\n includeFiles: true,\n includeTests: true,\n maxDigestLength: 400,\n};\n\n/**\n * Load sync options from config file\n */\nexport function loadSyncOptions(): SyncOptions {\n try {\n if (existsSync(SYNC_CONFIG_PATH)) {\n const data = JSON.parse(readFileSync(SYNC_CONFIG_PATH, 'utf8'));\n return parseConfigSafe(\n SyncOptionsSchema,\n { ...DEFAULT_SYNC_OPTIONS, ...data },\n DEFAULT_SYNC_OPTIONS,\n 'whatsapp-sync'\n );\n }\n } catch {\n // Use defaults\n }\n return { ...DEFAULT_SYNC_OPTIONS };\n}\n\n/**\n * Save sync options to config file\n */\nexport function saveSyncOptions(options: SyncOptions): void {\n try {\n ensureSecureDir(join(homedir(), '.stackmemory'));\n writeFileSecure(SYNC_CONFIG_PATH, JSON.stringify(options, null, 2));\n } catch {\n // Silently fail\n }\n}\n\n/**\n * Format duration for mobile display\n */\nfunction formatDuration(seconds: number): string {\n if (seconds < 60) {\n return `${seconds}s`;\n } else if (seconds < 3600) {\n const mins = Math.floor(seconds / 60);\n const secs = seconds % 60;\n return secs > 0 ? `${mins}m${secs}s` : `${mins}m`;\n } else {\n const hours = Math.floor(seconds / 3600);\n const mins = Math.floor((seconds % 3600) / 60);\n return mins > 0 ? `${hours}h${mins}m` : `${hours}h`;\n }\n}\n\n/**\n * Truncate text to max length with ellipsis\n */\nfunction truncate(text: string, maxLen: number): string {\n if (text.length <= maxLen) return text;\n return text.slice(0, maxLen - 3) + '...';\n}\n\n/**\n * Get status symbol for mobile display\n */\nfunction getStatusSymbol(status: string): string {\n switch (status) {\n case 'success':\n return 'OK';\n case 'failure':\n return 'FAIL';\n case 'partial':\n return 'PARTIAL';\n case 'ongoing':\n return 'ACTIVE';\n default:\n return '?';\n }\n}\n\n/**\n * Generate WhatsApp-friendly digest (300-400 chars max)\n * Optimized for mobile readability\n */\nexport function generateMobileDigest(\n data: FrameDigestData,\n options: SyncOptions = DEFAULT_SYNC_OPTIONS\n): string {\n const parts: string[] = [];\n const maxLen = options.maxDigestLength;\n\n // Header: FRAME: Name [type] - duration status\n const header = `FRAME: ${truncate(data.name, 30)} [${data.type}] - ${formatDuration(data.durationSeconds)} ${getStatusSymbol(data.status)}`;\n parts.push(header);\n\n // Activity summary line\n const activityParts: string[] = [];\n\n if (options.includeFiles && data.filesModified.length > 0) {\n activityParts.push(`FILES: ${data.filesModified.length}`);\n }\n\n if (data.toolCallCount > 0) {\n activityParts.push(`TOOLS: ${data.toolCallCount}`);\n }\n\n if (options.includeTests && data.testsRun.length > 0) {\n const passed = data.testsRun.filter((t) => t.status === 'passed').length;\n const failed = data.testsRun.filter((t) => t.status === 'failed').length;\n if (failed > 0) {\n activityParts.push(`TESTS: ${passed}ok/${failed}fail`);\n } else {\n activityParts.push(`TESTS: ${passed} pass`);\n }\n }\n\n if (activityParts.length > 0) {\n parts.push(activityParts.join(' | '));\n }\n\n // Files modified (compact)\n if (options.includeFiles && data.filesModified.length > 0) {\n const fileList = data.filesModified\n .slice(0, 3)\n .map((f) => {\n const basename = f.path.split('/').pop() || f.path;\n const op = f.operation.charAt(0).toUpperCase();\n return `${op}:${truncate(basename, 20)}`;\n })\n .join(', ');\n const more =\n data.filesModified.length > 3 ? ` +${data.filesModified.length - 3}` : '';\n parts.push(` ${fileList}${more}`);\n }\n\n // Decisions (high value)\n if (options.includeDecisions && data.decisions.length > 0) {\n parts.push('');\n parts.push('DECISIONS:');\n data.decisions.slice(0, 3).forEach((d) => {\n parts.push(` ${truncate(d, 60)}`);\n });\n if (data.decisions.length > 3) {\n parts.push(` +${data.decisions.length - 3} more`);\n }\n }\n\n // Risks (important to surface)\n if (data.risks.length > 0) {\n parts.push('');\n parts.push('RISKS:');\n data.risks.slice(0, 2).forEach((r) => {\n parts.push(` ${truncate(r, 50)}`);\n });\n }\n\n // Errors (unresolved only)\n const unresolvedErrors = data.errors.filter((e) => !e.resolved);\n if (unresolvedErrors.length > 0) {\n parts.push('');\n parts.push(`ERRORS: ${unresolvedErrors.length} unresolved`);\n unresolvedErrors.slice(0, 2).forEach((e) => {\n parts.push(` ${truncate(e.message, 50)}`);\n });\n }\n\n // Next action suggestion\n parts.push('');\n if (data.status === 'success') {\n parts.push('NEXT: commit & test');\n } else if (data.status === 'failure') {\n parts.push('NEXT: fix errors');\n } else if (data.status === 'partial') {\n parts.push('NEXT: review & continue');\n } else {\n parts.push('NEXT: check status');\n }\n\n // Join and truncate final result\n let result = parts.join('\\n');\n\n // If too long, trim aggressively\n if (result.length > maxLen) {\n // Remove less important sections\n const essentialParts = [header];\n\n if (activityParts.length > 0) {\n essentialParts.push(activityParts.join(' | '));\n }\n\n if (options.includeDecisions && data.decisions.length > 0) {\n essentialParts.push('');\n essentialParts.push(`DECISIONS: ${data.decisions.length}`);\n essentialParts.push(` ${truncate(data.decisions[0], 50)}`);\n }\n\n if (unresolvedErrors.length > 0) {\n essentialParts.push('');\n essentialParts.push(`ERRORS: ${unresolvedErrors.length} unresolved`);\n }\n\n essentialParts.push('');\n essentialParts.push(\n data.status === 'success' ? 'NEXT: commit' : 'NEXT: review'\n );\n\n result = essentialParts.join('\\n');\n }\n\n return result.slice(0, maxLen);\n}\n\n/**\n * Get frame digest data from stackmemory database\n * Returns null if frame not found or context unavailable\n */\nexport async function getFrameDigestData(\n frameId?: string\n): Promise<FrameDigestData | null> {\n try {\n // Try to load from recent frame digests\n const digestPath = join(\n homedir(),\n '.stackmemory',\n 'latest-frame-digest.json'\n );\n\n if (existsSync(digestPath)) {\n const data = JSON.parse(readFileSync(digestPath, 'utf8'));\n\n // If frameId specified, check if it matches\n if (frameId && data.frameId !== frameId) {\n return null;\n }\n\n return data as FrameDigestData;\n }\n\n return null;\n } catch {\n return null;\n }\n}\n\n/**\n * Store frame digest for sync\n */\nexport function storeFrameDigest(data: FrameDigestData): void {\n try {\n ensureSecureDir(join(homedir(), '.stackmemory'));\n const digestPath = join(\n homedir(),\n '.stackmemory',\n 'latest-frame-digest.json'\n );\n writeFileSecure(digestPath, JSON.stringify(data, null, 2));\n } catch {\n // Silently fail\n }\n}\n\n/**\n * Push current context to WhatsApp\n */\nexport async function syncContext(): Promise<SyncResult> {\n const options = loadSyncOptions();\n const data = await getFrameDigestData();\n\n if (!data) {\n return {\n success: false,\n error: 'No context data available. Run a task first.',\n };\n }\n\n return syncFrameData(data, options);\n}\n\n/**\n * Sync specific frame by ID\n */\nexport async function syncFrame(frameId: string): Promise<SyncResult> {\n const options = loadSyncOptions();\n const data = await getFrameDigestData(frameId);\n\n if (!data) {\n return {\n success: false,\n error: `Frame not found: ${frameId}`,\n };\n }\n\n return syncFrameData(data, options);\n}\n\n/**\n * Sync frame data to WhatsApp\n */\nasync function syncFrameData(\n data: FrameDigestData,\n options: SyncOptions\n): Promise<SyncResult> {\n const config = loadSMSConfig();\n\n if (!config.enabled) {\n return { success: false, error: 'Notifications disabled' };\n }\n\n // Check minimum duration\n if (data.durationSeconds < options.minFrameDuration) {\n return {\n success: false,\n error: `Frame too short (${data.durationSeconds}s < ${options.minFrameDuration}s min)`,\n };\n }\n\n // Generate mobile digest\n const digest = generateMobileDigest(data, options);\n\n // Send notification\n const payload: NotificationPayload = {\n type: 'custom',\n title: 'Context Sync',\n message: digest,\n prompt: {\n type: 'options',\n options: [\n { key: '1', label: 'Commit', action: 'git add -A && git commit' },\n { key: '2', label: 'Status', action: 'stackmemory status' },\n { key: '3', label: 'Continue', action: 'echo \"Continuing...\"' },\n ],\n question: 'Action?',\n },\n };\n\n const result = await sendNotification(payload);\n\n return {\n success: result.success,\n messageId: result.promptId,\n channel: result.channel,\n error: result.error,\n digestLength: digest.length,\n };\n}\n\n/**\n * Enable auto-sync on frame close\n */\nexport function enableAutoSync(options?: Partial<SyncOptions>): void {\n const current = loadSyncOptions();\n const updated: SyncOptions = {\n ...current,\n ...options,\n autoSyncOnClose: true,\n };\n saveSyncOptions(updated);\n\n // Update SMS config to enable context_sync notifications\n const smsConfig = loadSMSConfig();\n if (!smsConfig.notifyOn.custom) {\n smsConfig.notifyOn.custom = true;\n saveSMSConfig(smsConfig);\n }\n}\n\n/**\n * Disable auto-sync\n */\nexport function disableAutoSync(): void {\n const current = loadSyncOptions();\n current.autoSyncOnClose = false;\n saveSyncOptions(current);\n}\n\n/**\n * Check if auto-sync is enabled\n */\nexport function isAutoSyncEnabled(): boolean {\n const options = loadSyncOptions();\n return options.autoSyncOnClose;\n}\n\n/**\n * Callback for frame manager to trigger auto-sync\n * Call this when a frame closes\n */\nexport async function onFrameClosed(\n frameData: FrameDigestData\n): Promise<SyncResult | null> {\n if (!isAutoSyncEnabled()) {\n return null;\n }\n\n // Store for potential manual sync later\n storeFrameDigest(frameData);\n\n // Auto-sync\n const options = loadSyncOptions();\n return syncFrameData(frameData, options);\n}\n\n/**\n * Internal hook handler that receives FrameCloseData from lifecycle hooks\n */\nasync function handleFrameCloseHook(data: FrameCloseData): Promise<void> {\n const digestData = createFrameDigestData(\n data.frame,\n data.events,\n data.anchors\n );\n await onFrameClosed(digestData);\n}\n\n// Track if hook is registered to avoid duplicates\nlet hookUnregister: (() => void) | null = null;\n\n/**\n * Register WhatsApp sync as a frame lifecycle hook\n * This enables automatic sync when frames are closed\n * Call this during app initialization to enable the integration\n *\n * @returns Unregister function to disable the hook\n */\nexport function registerWhatsAppSyncHook(): () => void {\n // Avoid duplicate registration\n if (hookUnregister) {\n return hookUnregister;\n }\n\n hookUnregister = frameLifecycleHooks.onFrameClosed(\n 'whatsapp-sync',\n handleFrameCloseHook,\n -10 // Low priority - run after other hooks\n );\n\n return () => {\n if (hookUnregister) {\n hookUnregister();\n hookUnregister = null;\n }\n };\n}\n\n/**\n * Check if the WhatsApp sync hook is currently registered\n */\nexport function isHookRegistered(): boolean {\n return hookUnregister !== null;\n}\n\n/**\n * Create frame digest data from raw frame info\n * Helper for integration with frame-manager\n */\nexport function createFrameDigestData(\n frame: {\n frame_id: string;\n name: string;\n type: string;\n created_at: number;\n closed_at?: number;\n },\n events: Array<{\n event_type: string;\n payload: Record<string, unknown>;\n }>,\n anchors: Array<{\n type: string;\n text: string;\n }>\n): FrameDigestData {\n const now = Math.floor(Date.now() / 1000);\n const duration = (frame.closed_at || now) - frame.created_at;\n\n // Extract files from tool_call events\n const filesModified: Array<{ path: string; operation: string }> = [];\n const filesSeen = new Set<string>();\n\n events\n .filter((e) => e.event_type === 'tool_call')\n .forEach((e) => {\n const path = e.payload['path'] as string | undefined;\n if (path && !filesSeen.has(path)) {\n filesSeen.add(path);\n const toolName = (e.payload['tool_name'] as string) || '';\n let operation = 'modify';\n if (toolName.includes('Write') || toolName.includes('Create')) {\n operation = 'create';\n } else if (toolName.includes('Read')) {\n operation = 'read';\n } else if (toolName.includes('Delete')) {\n operation = 'delete';\n }\n filesModified.push({ path, operation });\n }\n });\n\n // Extract tests\n const testsRun: Array<{ name: string; status: string }> = [];\n events\n .filter(\n (e) =>\n e.event_type === 'tool_result' &&\n String(e.payload['output'] || '').includes('test')\n )\n .forEach((e) => {\n const output = String(e.payload['output'] || '');\n // Simple test result extraction\n const passMatch = output.match(/(\\d+) pass/i);\n const failMatch = output.match(/(\\d+) fail/i);\n if (passMatch) {\n testsRun.push({ name: 'Tests', status: 'passed' });\n }\n if (failMatch && parseInt(failMatch[1]) > 0) {\n testsRun.push({ name: 'Tests', status: 'failed' });\n }\n });\n\n // Extract decisions and risks from anchors\n const decisions = anchors\n .filter((a) => a.type === 'DECISION')\n .map((a) => a.text);\n\n const risks = anchors.filter((a) => a.type === 'RISK').map((a) => a.text);\n\n // Extract errors\n const errors: Array<{ type: string; message: string; resolved: boolean }> =\n [];\n events\n .filter((e) => e.payload['error'] || e.payload['status'] === 'error')\n .forEach((e) => {\n const errorMsg =\n (e.payload['error'] as string) ||\n (e.payload['message'] as string) ||\n 'Unknown error';\n errors.push({\n type: (e.payload['type'] as string) || 'error',\n message: errorMsg,\n resolved: false,\n });\n });\n\n // Determine status\n let status: 'success' | 'failure' | 'partial' | 'ongoing' = 'ongoing';\n if (frame.closed_at) {\n if (errors.filter((e) => !e.resolved).length > 0) {\n status = 'failure';\n } else if (\n testsRun.some((t) => t.status === 'failed') ||\n filesModified.length === 0\n ) {\n status = 'partial';\n } else {\n status = 'success';\n }\n }\n\n // Count tool calls\n const toolCallCount = events.filter(\n (e) => e.event_type === 'tool_call'\n ).length;\n\n return {\n frameId: frame.frame_id,\n name: frame.name,\n type: frame.type,\n status,\n durationSeconds: duration,\n filesModified: filesModified.filter((f) => f.operation !== 'read'),\n testsRun,\n decisions,\n risks,\n toolCallCount,\n errors,\n };\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": []
|
|
7
|
+
}
|
package/dist/index.js
CHANGED
|
@@ -4,7 +4,7 @@ const __filename = __fileURLToPath(import.meta.url);
|
|
|
4
4
|
const __dirname = __pathDirname(__filename);
|
|
5
5
|
import {
|
|
6
6
|
FrameManager
|
|
7
|
-
} from "./core/context/
|
|
7
|
+
} from "./core/context/index.js";
|
|
8
8
|
import { logger, Logger, LogLevel } from "./core/monitoring/logger.js";
|
|
9
9
|
import {
|
|
10
10
|
StackMemoryError,
|
package/dist/index.js.map
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../src/index.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * StackMemory - Lossless memory runtime for AI coding tools\n * Main entry point for the StackMemory package\n */\n\nexport {\n FrameManager,\n type FrameType,\n type FrameState,\n} from './core/context/
|
|
4
|
+
"sourcesContent": ["/**\n * StackMemory - Lossless memory runtime for AI coding tools\n * Main entry point for the StackMemory package\n */\n\nexport {\n FrameManager,\n type FrameType,\n type FrameState,\n} from './core/context/index.js';\nexport { logger, Logger, LogLevel } from './core/monitoring/logger.js';\nexport {\n StackMemoryError,\n ErrorCode,\n ErrorHandler,\n} from './core/monitoring/error-handler.js';\nexport { default as LocalStackMemoryMCP } from './integrations/mcp/server.js';\n\n// Re-export key types\nexport interface StackMemoryConfig {\n projectRoot?: string;\n dbPath?: string;\n logLevel?: 'ERROR' | 'WARN' | 'INFO' | 'DEBUG';\n}\n\nexport interface ContextItem {\n id: string;\n type: string;\n content: string;\n importance: number;\n timestamp: number;\n}\n"],
|
|
5
5
|
"mappings": ";;;;AAKA;AAAA,EACE;AAAA,OAGK;AACP,SAAS,QAAQ,QAAQ,gBAAgB;AACzC;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAoB,WAAXA,gBAAsC;",
|
|
6
6
|
"names": ["default"]
|
|
7
7
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/integrations/mcp/handlers/context-handlers.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * Context-related MCP tool handlers\n * Handles frame management and context retrieval\n */\n\nimport { FrameManager, FrameType } from '../../../core/context/
|
|
4
|
+
"sourcesContent": ["/**\n * Context-related MCP tool handlers\n * Handles frame management and context retrieval\n */\n\nimport { FrameManager, FrameType } from '../../../core/context/index.js';\nimport { LLMContextRetrieval } from '../../../core/retrieval/index.js';\nimport { logger } from '../../../core/monitoring/logger.js';\n\nexport interface ContextHandlerDependencies {\n frameManager: FrameManager;\n contextRetrieval: LLMContextRetrieval;\n}\n\nexport class ContextHandlers {\n constructor(private deps: ContextHandlerDependencies) {}\n\n /**\n * Get current project context\n */\n async handleGetContext(args: any): Promise<any> {\n try {\n const query = args.query || '';\n const limit = args.limit || 5;\n\n logger.info('Getting context', { query, limit });\n\n // Get hot stack context\n const hotStack = this.deps.frameManager.getHotStackContext(20);\n\n if (hotStack.length === 0) {\n return {\n content: [\n {\n type: 'text',\n text: 'No active context frames found. Use start_frame to begin working on a task.',\n },\n ],\n };\n }\n\n // Use LLM context retrieval if available\n if (this.deps.contextRetrieval && query) {\n try {\n // TODO: Implement getRelevantContext method\n const llmContext = { summary: 'Context retrieval not yet implemented', frameIds: [] };\n return {\n content: [\n {\n type: 'text',\n text: llmContext.summary || 'No specific context found.',\n },\n ],\n metadata: {\n relevantFrames: llmContext.frameIds,\n query,\n },\n };\n } catch (error: unknown) {\n logger.warn('LLM context retrieval failed, falling back to hot stack', error instanceof Error ? error : new Error(String(error)));\n }\n }\n\n // Format hot stack context\n const contextText = hotStack\n .map((frame, i) => {\n const depth = ' '.repeat(i);\n const constraints = frame.header.constraints?.length\n ? `\\n${depth} Constraints: ${frame.header.constraints.join(', ')}`\n : '';\n const events = frame.recentEvents.length\n ? `\\n${depth} Recent: ${frame.recentEvents.length} events`\n : '';\n\n return `${depth}Frame ${i + 1}: ${frame.header.goal}${constraints}${events}`;\n })\n .join('\\n');\n\n return {\n content: [\n {\n type: 'text',\n text: `Current Context Stack:\\n${contextText}`,\n },\n ],\n };\n } catch (error: unknown) {\n logger.error('Error getting context', error instanceof Error ? error : new Error(String(error)));\n throw error;\n }\n }\n\n /**\n * Record a decision or important information\n */\n async handleAddDecision(args: any): Promise<any> {\n try {\n const { content, type } = args;\n \n if (!content) {\n throw new Error('Content is required');\n }\n\n const currentFrame = this.deps.frameManager.getCurrentFrameId();\n if (!currentFrame) {\n throw new Error('No active frame. Use start_frame first.');\n }\n\n // Add as anchor\n this.deps.frameManager.addAnchor(\n type === 'constraint' ? 'CONSTRAINT' : 'DECISION',\n content,\n type === 'constraint' ? 9 : 7\n );\n\n // Also add as event\n this.deps.frameManager.addEvent('decision', {\n type,\n content,\n timestamp: Date.now(),\n });\n\n logger.info('Added decision/constraint', { type, content });\n\n return {\n content: [\n {\n type: 'text',\n text: `Recorded ${type}: ${content}`,\n },\n ],\n };\n } catch (error: unknown) {\n logger.error('Error adding decision', error instanceof Error ? error : new Error(String(error)));\n throw error;\n }\n }\n\n /**\n * Start a new frame (task/subtask) on the call stack\n */\n async handleStartFrame(args: any): Promise<any> {\n try {\n const { name, type = 'task', constraints = [], definitions = {} } = args;\n \n if (!name) {\n throw new Error('Frame name is required');\n }\n\n const frameId = this.deps.frameManager.createFrame({\n type: type as FrameType,\n name,\n inputs: { constraints, definitions }\n });\n\n logger.info('Started frame', { frameId, name, type });\n\n return {\n content: [\n {\n type: 'text',\n text: `Started frame: ${name} (${frameId})`,\n },\n ],\n metadata: {\n frameId,\n type,\n name,\n },\n };\n } catch (error: unknown) {\n logger.error('Error starting frame', error instanceof Error ? error : new Error(String(error)));\n throw error;\n }\n }\n\n /**\n * Close current frame with summary\n */\n async handleCloseFrame(args: any): Promise<any> {\n try {\n const { summary, frameId } = args;\n const targetFrameId = frameId || this.deps.frameManager.getCurrentFrameId();\n\n if (!targetFrameId) {\n throw new Error('No active frame to close');\n }\n\n const frame = this.deps.frameManager.getFrame(targetFrameId);\n if (!frame) {\n throw new Error(`Frame not found: ${targetFrameId}`);\n }\n\n // Add summary if provided\n if (summary) {\n this.deps.frameManager.addEvent('observation', {\n type: 'completion_summary',\n content: summary,\n timestamp: Date.now(),\n });\n }\n\n this.deps.frameManager.closeFrame(targetFrameId, summary ? { summary } : {});\n\n logger.info('Closed frame', { frameId: targetFrameId, frameName: frame.name });\n\n return {\n content: [\n {\n type: 'text',\n text: `Closed frame: ${frame.name}${summary ? ` with summary: ${summary}` : ''}`,\n },\n ],\n };\n } catch (error: unknown) {\n logger.error('Error closing frame', error instanceof Error ? error : new Error(String(error)));\n throw error;\n }\n }\n\n /**\n * Add an anchor (important fact) to current frame\n */\n async handleAddAnchor(args: any): Promise<any> {\n try {\n const { type, text, priority = 5 } = args;\n \n if (!text) {\n throw new Error('Anchor text is required');\n }\n\n const currentFrame = this.deps.frameManager.getCurrentFrameId();\n if (!currentFrame) {\n throw new Error('No active frame. Use start_frame first.');\n }\n\n const validTypes = ['FACT', 'DECISION', 'CONSTRAINT', 'INTERFACE_CONTRACT', 'TODO', 'RISK'];\n if (!validTypes.includes(type)) {\n throw new Error(`Invalid anchor type. Must be one of: ${validTypes.join(', ')}`);\n }\n\n this.deps.frameManager.addAnchor(type, text, priority);\n\n logger.info('Added anchor', { type, text, priority });\n\n return {\n content: [\n {\n type: 'text',\n text: `Added ${type.toLowerCase()}: ${text}`,\n },\n ],\n };\n } catch (error: unknown) {\n logger.error('Error adding anchor', error instanceof Error ? error : new Error(String(error)));\n throw error;\n }\n }\n\n /**\n * Get current hot stack context\n */\n async handleGetHotStack(args: any): Promise<any> {\n try {\n const maxEvents = args.max_events || 10;\n const hotStack = this.deps.frameManager.getHotStackContext(maxEvents);\n\n if (hotStack.length === 0) {\n return {\n content: [\n {\n type: 'text',\n text: 'No active frames on the stack.',\n },\n ],\n };\n }\n\n const stackSummary = hotStack.map((frame, index) => ({\n depth: index,\n frameId: frame.frameId,\n goal: frame.header.goal,\n constraints: frame.header.constraints || [],\n anchors: frame.anchors.length,\n recentEvents: frame.recentEvents.length,\n artifacts: frame.activeArtifacts.length,\n }));\n\n return {\n content: [\n {\n type: 'text',\n text: `Hot Stack (${hotStack.length} frames):\\n` +\n stackSummary.map((f: any) => \n ` ${f.depth}: ${f.goal} (${f.anchors} anchors, ${f.recentEvents} events)`\n ).join('\\n'),\n },\n ],\n metadata: {\n stack: stackSummary,\n },\n };\n } catch (error: unknown) {\n logger.error('Error getting hot stack', error instanceof Error ? error : new Error(String(error)));\n throw error;\n }\n }\n}"],
|
|
5
5
|
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|