@stackmemoryai/stackmemory 0.5.30 → 0.5.33

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 (64) hide show
  1. package/README.md +44 -44
  2. package/dist/cli/claude-sm.js +199 -16
  3. package/dist/cli/claude-sm.js.map +2 -2
  4. package/dist/cli/commands/context.js +0 -11
  5. package/dist/cli/commands/context.js.map +2 -2
  6. package/dist/cli/commands/linear.js +1 -14
  7. package/dist/cli/commands/linear.js.map +2 -2
  8. package/dist/cli/commands/login.js +32 -10
  9. package/dist/cli/commands/login.js.map +2 -2
  10. package/dist/cli/commands/migrate.js +80 -22
  11. package/dist/cli/commands/migrate.js.map +2 -2
  12. package/dist/cli/commands/model.js +533 -0
  13. package/dist/cli/commands/model.js.map +7 -0
  14. package/dist/cli/commands/ralph.js +93 -28
  15. package/dist/cli/commands/ralph.js.map +2 -2
  16. package/dist/cli/commands/service.js +10 -3
  17. package/dist/cli/commands/service.js.map +2 -2
  18. package/dist/cli/commands/skills.js +60 -10
  19. package/dist/cli/commands/skills.js.map +2 -2
  20. package/dist/cli/commands/sms-notify.js +342 -22
  21. package/dist/cli/commands/sms-notify.js.map +3 -3
  22. package/dist/cli/index.js +2 -0
  23. package/dist/cli/index.js.map +2 -2
  24. package/dist/core/context/dual-stack-manager.js +23 -7
  25. package/dist/core/context/dual-stack-manager.js.map +2 -2
  26. package/dist/core/context/frame-database.js +33 -5
  27. package/dist/core/context/frame-database.js.map +2 -2
  28. package/dist/core/context/frame-digest.js +6 -1
  29. package/dist/core/context/frame-digest.js.map +2 -2
  30. package/dist/core/context/frame-manager.js +56 -9
  31. package/dist/core/context/frame-manager.js.map +2 -2
  32. package/dist/core/context/permission-manager.js +0 -11
  33. package/dist/core/context/permission-manager.js.map +2 -2
  34. package/dist/core/context/recursive-context-manager.js +15 -9
  35. package/dist/core/context/recursive-context-manager.js.map +2 -2
  36. package/dist/core/context/shared-context-layer.js +0 -11
  37. package/dist/core/context/shared-context-layer.js.map +2 -2
  38. package/dist/core/context/validation.js +6 -1
  39. package/dist/core/context/validation.js.map +2 -2
  40. package/dist/core/models/fallback-monitor.js +229 -0
  41. package/dist/core/models/fallback-monitor.js.map +7 -0
  42. package/dist/core/models/model-router.js +331 -0
  43. package/dist/core/models/model-router.js.map +7 -0
  44. package/dist/hooks/claude-code-whatsapp-hook.js +197 -0
  45. package/dist/hooks/claude-code-whatsapp-hook.js.map +7 -0
  46. package/dist/hooks/linear-task-picker.js +1 -1
  47. package/dist/hooks/linear-task-picker.js.map +2 -2
  48. package/dist/hooks/schemas.js +55 -1
  49. package/dist/hooks/schemas.js.map +2 -2
  50. package/dist/hooks/session-summary.js +5 -1
  51. package/dist/hooks/session-summary.js.map +2 -2
  52. package/dist/hooks/sms-action-runner.js +12 -1
  53. package/dist/hooks/sms-action-runner.js.map +2 -2
  54. package/dist/hooks/sms-notify.js +4 -2
  55. package/dist/hooks/sms-notify.js.map +2 -2
  56. package/dist/hooks/sms-webhook.js +23 -2
  57. package/dist/hooks/sms-webhook.js.map +2 -2
  58. package/dist/hooks/whatsapp-commands.js +376 -0
  59. package/dist/hooks/whatsapp-commands.js.map +7 -0
  60. package/dist/hooks/whatsapp-scheduler.js +317 -0
  61. package/dist/hooks/whatsapp-scheduler.js.map +7 -0
  62. package/dist/hooks/whatsapp-sync.js +375 -0
  63. package/dist/hooks/whatsapp-sync.js.map +7 -0
  64. package/package.json +2 -3
@@ -0,0 +1,376 @@
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 { writeFileSecure, ensureSecureDir } from "./secure-fs.js";
9
+ import { WhatsAppCommandsConfigSchema, parseConfigSafe } from "./schemas.js";
10
+ import { executeActionSafe } from "./sms-action-runner.js";
11
+ import {
12
+ syncContext,
13
+ getFrameDigestData,
14
+ generateMobileDigest,
15
+ loadSyncOptions
16
+ } from "./whatsapp-sync.js";
17
+ import { sendNotification } from "./sms-notify.js";
18
+ const CONFIG_PATH = join(homedir(), ".stackmemory", "whatsapp-commands.json");
19
+ const DEFAULT_COMMANDS = [
20
+ {
21
+ name: "status",
22
+ description: "Get current task/frame status",
23
+ enabled: true
24
+ // No action - handled specially in-process
25
+ },
26
+ {
27
+ name: "tasks",
28
+ description: "List active tasks",
29
+ enabled: true
30
+ // No action - handled specially in-process
31
+ },
32
+ {
33
+ name: "context",
34
+ description: "Get latest context digest",
35
+ enabled: true
36
+ // No action - handled specially
37
+ },
38
+ {
39
+ name: "approve",
40
+ description: "Approve a PR (requires PR number)",
41
+ enabled: true,
42
+ requiresArg: true,
43
+ argPattern: "^\\d+$"
44
+ // PR number must be numeric
45
+ },
46
+ {
47
+ name: "merge",
48
+ description: "Merge a PR (requires PR number)",
49
+ enabled: true,
50
+ requiresArg: true,
51
+ argPattern: "^\\d+$"
52
+ },
53
+ {
54
+ name: "help",
55
+ description: "List available commands",
56
+ enabled: true
57
+ // No action - handled specially
58
+ },
59
+ {
60
+ name: "sync",
61
+ description: "Push current context to WhatsApp",
62
+ enabled: true
63
+ // No action - handled specially
64
+ },
65
+ {
66
+ name: "build",
67
+ description: "Run npm build",
68
+ enabled: true,
69
+ action: "npm run build"
70
+ },
71
+ {
72
+ name: "test",
73
+ description: "Run tests",
74
+ enabled: true,
75
+ action: "npm run test:run"
76
+ },
77
+ {
78
+ name: "lint",
79
+ description: "Run linter",
80
+ enabled: true,
81
+ action: "npm run lint"
82
+ },
83
+ {
84
+ name: "log",
85
+ description: "Show recent git commits",
86
+ enabled: true,
87
+ action: "git log --oneline -5"
88
+ },
89
+ {
90
+ name: "diff",
91
+ description: "Show git diff summary",
92
+ enabled: true,
93
+ action: "git diff --stat"
94
+ },
95
+ {
96
+ name: "pr",
97
+ description: "List open PRs",
98
+ enabled: true,
99
+ action: "gh pr list"
100
+ },
101
+ {
102
+ name: "branch",
103
+ description: "Show current branch",
104
+ enabled: true,
105
+ action: "git branch --show-current"
106
+ }
107
+ ];
108
+ const DEFAULT_CONFIG = {
109
+ enabled: true,
110
+ commands: DEFAULT_COMMANDS
111
+ };
112
+ function loadCommandsConfig() {
113
+ try {
114
+ if (existsSync(CONFIG_PATH)) {
115
+ const data = JSON.parse(readFileSync(CONFIG_PATH, "utf8"));
116
+ return parseConfigSafe(
117
+ WhatsAppCommandsConfigSchema,
118
+ { ...DEFAULT_CONFIG, ...data },
119
+ DEFAULT_CONFIG,
120
+ "whatsapp-commands"
121
+ );
122
+ }
123
+ } catch {
124
+ }
125
+ return { ...DEFAULT_CONFIG };
126
+ }
127
+ function saveCommandsConfig(config) {
128
+ try {
129
+ ensureSecureDir(join(homedir(), ".stackmemory"));
130
+ writeFileSecure(CONFIG_PATH, JSON.stringify(config, null, 2));
131
+ } catch {
132
+ }
133
+ }
134
+ function isCommand(message) {
135
+ const trimmed = message.trim().toLowerCase();
136
+ const config = loadCommandsConfig();
137
+ if (!config.enabled) return false;
138
+ const words = trimmed.split(/\s+/);
139
+ const firstWord = words[0];
140
+ return config.commands.some(
141
+ (cmd) => cmd.enabled && cmd.name.toLowerCase() === firstWord
142
+ );
143
+ }
144
+ function parseCommand(message) {
145
+ const trimmed = message.trim();
146
+ const words = trimmed.split(/\s+/);
147
+ if (words.length === 0) return null;
148
+ const name = words[0].toLowerCase();
149
+ const arg = words.slice(1).join(" ").trim() || void 0;
150
+ return { name, arg };
151
+ }
152
+ function generateHelpText(config) {
153
+ const lines = ["Available commands:"];
154
+ config.commands.filter((cmd) => cmd.enabled).forEach((cmd) => {
155
+ const argHint = cmd.requiresArg ? " <arg>" : "";
156
+ lines.push(` ${cmd.name}${argHint} - ${cmd.description}`);
157
+ });
158
+ lines.push("");
159
+ lines.push("Reply with command name to execute");
160
+ return lines.join("\n");
161
+ }
162
+ async function handleContextCommand() {
163
+ const data = await getFrameDigestData();
164
+ if (!data) {
165
+ return "No context available. Start a task first.";
166
+ }
167
+ const options = loadSyncOptions();
168
+ return generateMobileDigest(data, options);
169
+ }
170
+ async function handleSyncCommand() {
171
+ const result = await syncContext();
172
+ if (result.success) {
173
+ return `Context synced (${result.digestLength} chars)`;
174
+ } else {
175
+ return `Sync failed: ${result.error}`;
176
+ }
177
+ }
178
+ async function handleStatusCommand() {
179
+ try {
180
+ const data = await getFrameDigestData();
181
+ if (!data) {
182
+ return "No active session. Start with: claude-sm";
183
+ }
184
+ const lines = [];
185
+ lines.push(`Frame: ${data.name || data.frameId}`);
186
+ lines.push(`Status: ${data.status}`);
187
+ lines.push(`Files: ${data.filesModified?.length || 0} modified`);
188
+ lines.push(`Tools: ${data.toolCallCount || 0} calls`);
189
+ if (data.errors?.length > 0) {
190
+ const unresolved = data.errors.filter((e) => !e.resolved).length;
191
+ if (unresolved > 0) lines.push(`Errors: ${unresolved} unresolved`);
192
+ }
193
+ lines.push(`Duration: ${Math.round(data.durationSeconds / 60)}min`);
194
+ return lines.join("\n");
195
+ } catch {
196
+ return "Status unavailable";
197
+ }
198
+ }
199
+ async function handleTasksCommand() {
200
+ try {
201
+ const data = await getFrameDigestData();
202
+ if (!data) {
203
+ return "No active tasks";
204
+ }
205
+ const lines = [];
206
+ if (data.decisions?.length > 0) {
207
+ lines.push("Recent decisions:");
208
+ data.decisions.slice(0, 3).forEach((d, i) => {
209
+ lines.push(
210
+ `${i + 1}. ${d.substring(0, 50)}${d.length > 50 ? "..." : ""}`
211
+ );
212
+ });
213
+ }
214
+ if (data.risks?.length > 0) {
215
+ lines.push("");
216
+ lines.push("Risks:");
217
+ data.risks.slice(0, 2).forEach((r) => {
218
+ lines.push(`- ${r.substring(0, 50)}${r.length > 50 ? "..." : ""}`);
219
+ });
220
+ }
221
+ if (lines.length === 0) {
222
+ return "No active tasks or decisions";
223
+ }
224
+ return lines.join("\n");
225
+ } catch {
226
+ return "Tasks unavailable";
227
+ }
228
+ }
229
+ async function processCommand(from, message) {
230
+ const config = loadCommandsConfig();
231
+ if (!config.enabled) {
232
+ return { handled: false };
233
+ }
234
+ const parsed = parseCommand(message);
235
+ if (!parsed) {
236
+ return { handled: false };
237
+ }
238
+ const command = config.commands.find(
239
+ (cmd) => cmd.enabled && cmd.name.toLowerCase() === parsed.name
240
+ );
241
+ if (!command) {
242
+ return { handled: false };
243
+ }
244
+ if (command.name === "help") {
245
+ const helpText = generateHelpText(config);
246
+ return { handled: true, response: helpText };
247
+ }
248
+ if (command.name === "context") {
249
+ const contextText = await handleContextCommand();
250
+ return { handled: true, response: contextText };
251
+ }
252
+ if (command.name === "sync") {
253
+ const syncText = await handleSyncCommand();
254
+ return { handled: true, response: syncText };
255
+ }
256
+ if (command.name === "status") {
257
+ const statusText = await handleStatusCommand();
258
+ return { handled: true, response: statusText };
259
+ }
260
+ if (command.name === "tasks") {
261
+ const tasksText = await handleTasksCommand();
262
+ return { handled: true, response: tasksText };
263
+ }
264
+ if (command.requiresArg && !parsed.arg) {
265
+ return {
266
+ handled: true,
267
+ response: `${command.name} requires an argument. Usage: ${command.name} <arg>`,
268
+ error: "Missing argument"
269
+ };
270
+ }
271
+ if (command.argPattern && parsed.arg) {
272
+ const pattern = new RegExp(command.argPattern);
273
+ if (!pattern.test(parsed.arg)) {
274
+ return {
275
+ handled: true,
276
+ response: `Invalid argument format for ${command.name}`,
277
+ error: "Invalid argument format"
278
+ };
279
+ }
280
+ }
281
+ let action = command.action;
282
+ if (action && parsed.arg) {
283
+ if (command.name === "approve") {
284
+ action = `gh pr review ${parsed.arg} --approve`;
285
+ } else if (command.name === "merge") {
286
+ action = `gh pr merge ${parsed.arg} --squash`;
287
+ }
288
+ }
289
+ if (action) {
290
+ console.log(`[whatsapp-commands] Executing: ${action}`);
291
+ const result = await executeActionSafe(action, message);
292
+ if (result.success) {
293
+ const output = result.output?.slice(0, 200) || "Done";
294
+ return {
295
+ handled: true,
296
+ response: `${command.name}: ${output}`,
297
+ action
298
+ };
299
+ } else {
300
+ return {
301
+ handled: true,
302
+ response: `${command.name} failed: ${result.error?.slice(0, 100)}`,
303
+ error: result.error,
304
+ action
305
+ };
306
+ }
307
+ }
308
+ return {
309
+ handled: true,
310
+ response: `Command ${command.name} acknowledged`
311
+ };
312
+ }
313
+ async function sendCommandResponse(response) {
314
+ const result = await sendNotification({
315
+ type: "custom",
316
+ title: "Command Result",
317
+ message: response
318
+ });
319
+ return { success: result.success, error: result.error };
320
+ }
321
+ function enableCommands() {
322
+ const config = loadCommandsConfig();
323
+ config.enabled = true;
324
+ saveCommandsConfig(config);
325
+ }
326
+ function disableCommands() {
327
+ const config = loadCommandsConfig();
328
+ config.enabled = false;
329
+ saveCommandsConfig(config);
330
+ }
331
+ function isCommandsEnabled() {
332
+ const config = loadCommandsConfig();
333
+ return config.enabled;
334
+ }
335
+ function addCommand(command) {
336
+ const config = loadCommandsConfig();
337
+ const existingIndex = config.commands.findIndex(
338
+ (c) => c.name.toLowerCase() === command.name.toLowerCase()
339
+ );
340
+ if (existingIndex >= 0) {
341
+ config.commands[existingIndex] = command;
342
+ } else {
343
+ config.commands.push(command);
344
+ }
345
+ saveCommandsConfig(config);
346
+ }
347
+ function removeCommand(name) {
348
+ const config = loadCommandsConfig();
349
+ const initialLength = config.commands.length;
350
+ config.commands = config.commands.filter(
351
+ (c) => c.name.toLowerCase() !== name.toLowerCase()
352
+ );
353
+ if (config.commands.length < initialLength) {
354
+ saveCommandsConfig(config);
355
+ return true;
356
+ }
357
+ return false;
358
+ }
359
+ function getAvailableCommands() {
360
+ const config = loadCommandsConfig();
361
+ return config.commands.filter((c) => c.enabled);
362
+ }
363
+ export {
364
+ addCommand,
365
+ disableCommands,
366
+ enableCommands,
367
+ getAvailableCommands,
368
+ isCommand,
369
+ isCommandsEnabled,
370
+ loadCommandsConfig,
371
+ processCommand,
372
+ removeCommand,
373
+ saveCommandsConfig,
374
+ sendCommandResponse
375
+ };
376
+ //# sourceMappingURL=whatsapp-commands.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/hooks/whatsapp-commands.ts"],
4
+ "sourcesContent": ["/**\n * WhatsApp Inbound Command Processor\n * Process WhatsApp messages as commands\n */\n\nimport { existsSync, readFileSync } from 'fs';\nimport { join } from 'path';\nimport { homedir } from 'os';\nimport { writeFileSecure, ensureSecureDir } from './secure-fs.js';\nimport { WhatsAppCommandsConfigSchema, parseConfigSafe } from './schemas.js';\nimport { executeActionSafe } from './sms-action-runner.js';\nimport {\n syncContext,\n getFrameDigestData,\n generateMobileDigest,\n loadSyncOptions,\n} from './whatsapp-sync.js';\nimport { sendNotification } from './sms-notify.js';\n\nexport interface WhatsAppCommand {\n name: string;\n description: string;\n enabled: boolean;\n action?: string; // Safe action to execute\n requiresArg?: boolean;\n argPattern?: string; // Regex pattern for arg validation\n}\n\nexport interface CommandsConfig {\n enabled: boolean;\n commands: WhatsAppCommand[];\n}\n\nexport interface CommandResult {\n handled: boolean;\n response?: string;\n action?: string;\n error?: string;\n}\n\nconst CONFIG_PATH = join(homedir(), '.stackmemory', 'whatsapp-commands.json');\n\n// Default supported commands\nconst DEFAULT_COMMANDS: WhatsAppCommand[] = [\n {\n name: 'status',\n description: 'Get current task/frame status',\n enabled: true,\n // No action - handled specially in-process\n },\n {\n name: 'tasks',\n description: 'List active tasks',\n enabled: true,\n // No action - handled specially in-process\n },\n {\n name: 'context',\n description: 'Get latest context digest',\n enabled: true,\n // No action - handled specially\n },\n {\n name: 'approve',\n description: 'Approve a PR (requires PR number)',\n enabled: true,\n requiresArg: true,\n argPattern: '^\\\\d+$', // PR number must be numeric\n },\n {\n name: 'merge',\n description: 'Merge a PR (requires PR number)',\n enabled: true,\n requiresArg: true,\n argPattern: '^\\\\d+$',\n },\n {\n name: 'help',\n description: 'List available commands',\n enabled: true,\n // No action - handled specially\n },\n {\n name: 'sync',\n description: 'Push current context to WhatsApp',\n enabled: true,\n // No action - handled specially\n },\n {\n name: 'build',\n description: 'Run npm build',\n enabled: true,\n action: 'npm run build',\n },\n {\n name: 'test',\n description: 'Run tests',\n enabled: true,\n action: 'npm run test:run',\n },\n {\n name: 'lint',\n description: 'Run linter',\n enabled: true,\n action: 'npm run lint',\n },\n {\n name: 'log',\n description: 'Show recent git commits',\n enabled: true,\n action: 'git log --oneline -5',\n },\n {\n name: 'diff',\n description: 'Show git diff summary',\n enabled: true,\n action: 'git diff --stat',\n },\n {\n name: 'pr',\n description: 'List open PRs',\n enabled: true,\n action: 'gh pr list',\n },\n {\n name: 'branch',\n description: 'Show current branch',\n enabled: true,\n action: 'git branch --show-current',\n },\n];\n\nconst DEFAULT_CONFIG: CommandsConfig = {\n enabled: true,\n commands: DEFAULT_COMMANDS,\n};\n\n/**\n * Load commands config\n */\nexport function loadCommandsConfig(): CommandsConfig {\n try {\n if (existsSync(CONFIG_PATH)) {\n const data = JSON.parse(readFileSync(CONFIG_PATH, 'utf8'));\n return parseConfigSafe(\n WhatsAppCommandsConfigSchema,\n { ...DEFAULT_CONFIG, ...data },\n DEFAULT_CONFIG,\n 'whatsapp-commands'\n );\n }\n } catch {\n // Use defaults\n }\n return { ...DEFAULT_CONFIG };\n}\n\n/**\n * Save commands config\n */\nexport function saveCommandsConfig(config: CommandsConfig): void {\n try {\n ensureSecureDir(join(homedir(), '.stackmemory'));\n writeFileSecure(CONFIG_PATH, JSON.stringify(config, null, 2));\n } catch {\n // Silently fail\n }\n}\n\n/**\n * Check if a message is a command\n */\nexport function isCommand(message: string): boolean {\n const trimmed = message.trim().toLowerCase();\n\n // Check if it's a single word command\n const config = loadCommandsConfig();\n if (!config.enabled) return false;\n\n const words = trimmed.split(/\\s+/);\n const firstWord = words[0];\n\n return config.commands.some(\n (cmd) => cmd.enabled && cmd.name.toLowerCase() === firstWord\n );\n}\n\n/**\n * Parse command from message\n */\nfunction parseCommand(message: string): { name: string; arg?: string } | null {\n const trimmed = message.trim();\n const words = trimmed.split(/\\s+/);\n\n if (words.length === 0) return null;\n\n const name = words[0].toLowerCase();\n const arg = words.slice(1).join(' ').trim() || undefined;\n\n return { name, arg };\n}\n\n/**\n * Generate help text for available commands\n */\nfunction generateHelpText(config: CommandsConfig): string {\n const lines: string[] = ['Available commands:'];\n\n config.commands\n .filter((cmd) => cmd.enabled)\n .forEach((cmd) => {\n const argHint = cmd.requiresArg ? ' <arg>' : '';\n lines.push(` ${cmd.name}${argHint} - ${cmd.description}`);\n });\n\n lines.push('');\n lines.push('Reply with command name to execute');\n\n return lines.join('\\n');\n}\n\n/**\n * Handle the 'context' command specially\n */\nasync function handleContextCommand(): Promise<string> {\n const data = await getFrameDigestData();\n\n if (!data) {\n return 'No context available. Start a task first.';\n }\n\n const options = loadSyncOptions();\n return generateMobileDigest(data, options);\n}\n\n/**\n * Handle the 'sync' command specially\n */\nasync function handleSyncCommand(): Promise<string> {\n const result = await syncContext();\n\n if (result.success) {\n return `Context synced (${result.digestLength} chars)`;\n } else {\n return `Sync failed: ${result.error}`;\n }\n}\n\n/**\n * Handle the 'status' command - get current frame/task status\n */\nasync function handleStatusCommand(): Promise<string> {\n try {\n const data = await getFrameDigestData();\n if (!data) {\n return 'No active session. Start with: claude-sm';\n }\n\n const lines: string[] = [];\n lines.push(`Frame: ${data.name || data.frameId}`);\n lines.push(`Status: ${data.status}`);\n lines.push(`Files: ${data.filesModified?.length || 0} modified`);\n lines.push(`Tools: ${data.toolCallCount || 0} calls`);\n if (data.errors?.length > 0) {\n const unresolved = data.errors.filter((e) => !e.resolved).length;\n if (unresolved > 0) lines.push(`Errors: ${unresolved} unresolved`);\n }\n lines.push(`Duration: ${Math.round(data.durationSeconds / 60)}min`);\n\n return lines.join('\\n');\n } catch {\n return 'Status unavailable';\n }\n}\n\n/**\n * Handle the 'tasks' command - list recent decisions/risks\n */\nasync function handleTasksCommand(): Promise<string> {\n try {\n const data = await getFrameDigestData();\n if (!data) {\n return 'No active tasks';\n }\n\n const lines: string[] = [];\n\n if (data.decisions?.length > 0) {\n lines.push('Recent decisions:');\n data.decisions.slice(0, 3).forEach((d, i) => {\n lines.push(\n `${i + 1}. ${d.substring(0, 50)}${d.length > 50 ? '...' : ''}`\n );\n });\n }\n\n if (data.risks?.length > 0) {\n lines.push('');\n lines.push('Risks:');\n data.risks.slice(0, 2).forEach((r) => {\n lines.push(`- ${r.substring(0, 50)}${r.length > 50 ? '...' : ''}`);\n });\n }\n\n if (lines.length === 0) {\n return 'No active tasks or decisions';\n }\n\n return lines.join('\\n');\n } catch {\n return 'Tasks unavailable';\n }\n}\n\n/**\n * Process an incoming WhatsApp command\n */\nexport async function processCommand(\n from: string,\n message: string\n): Promise<CommandResult> {\n const config = loadCommandsConfig();\n\n if (!config.enabled) {\n return { handled: false };\n }\n\n const parsed = parseCommand(message);\n if (!parsed) {\n return { handled: false };\n }\n\n const command = config.commands.find(\n (cmd) => cmd.enabled && cmd.name.toLowerCase() === parsed.name\n );\n\n if (!command) {\n return { handled: false };\n }\n\n // Handle special commands\n if (command.name === 'help') {\n const helpText = generateHelpText(config);\n return { handled: true, response: helpText };\n }\n\n if (command.name === 'context') {\n const contextText = await handleContextCommand();\n return { handled: true, response: contextText };\n }\n\n if (command.name === 'sync') {\n const syncText = await handleSyncCommand();\n return { handled: true, response: syncText };\n }\n\n if (command.name === 'status') {\n const statusText = await handleStatusCommand();\n return { handled: true, response: statusText };\n }\n\n if (command.name === 'tasks') {\n const tasksText = await handleTasksCommand();\n return { handled: true, response: tasksText };\n }\n\n // Check if argument is required\n if (command.requiresArg && !parsed.arg) {\n return {\n handled: true,\n response: `${command.name} requires an argument. Usage: ${command.name} <arg>`,\n error: 'Missing argument',\n };\n }\n\n // Validate argument pattern if specified\n if (command.argPattern && parsed.arg) {\n const pattern = new RegExp(command.argPattern);\n if (!pattern.test(parsed.arg)) {\n return {\n handled: true,\n response: `Invalid argument format for ${command.name}`,\n error: 'Invalid argument format',\n };\n }\n }\n\n // Build the action command\n let action = command.action;\n\n if (action && parsed.arg) {\n // Special handling for PR commands\n if (command.name === 'approve') {\n action = `gh pr review ${parsed.arg} --approve`;\n } else if (command.name === 'merge') {\n action = `gh pr merge ${parsed.arg} --squash`;\n }\n }\n\n // Execute the action if defined\n if (action) {\n console.log(`[whatsapp-commands] Executing: ${action}`);\n\n const result = await executeActionSafe(action, message);\n\n if (result.success) {\n const output = result.output?.slice(0, 200) || 'Done';\n return {\n handled: true,\n response: `${command.name}: ${output}`,\n action,\n };\n } else {\n return {\n handled: true,\n response: `${command.name} failed: ${result.error?.slice(0, 100)}`,\n error: result.error,\n action,\n };\n }\n }\n\n return {\n handled: true,\n response: `Command ${command.name} acknowledged`,\n };\n}\n\n/**\n * Send command result back via WhatsApp\n */\nexport async function sendCommandResponse(\n response: string\n): Promise<{ success: boolean; error?: string }> {\n const result = await sendNotification({\n type: 'custom',\n title: 'Command Result',\n message: response,\n });\n\n return { success: result.success, error: result.error };\n}\n\n/**\n * Enable command processing\n */\nexport function enableCommands(): void {\n const config = loadCommandsConfig();\n config.enabled = true;\n saveCommandsConfig(config);\n}\n\n/**\n * Disable command processing\n */\nexport function disableCommands(): void {\n const config = loadCommandsConfig();\n config.enabled = false;\n saveCommandsConfig(config);\n}\n\n/**\n * Check if commands are enabled\n */\nexport function isCommandsEnabled(): boolean {\n const config = loadCommandsConfig();\n return config.enabled;\n}\n\n/**\n * Add a custom command\n */\nexport function addCommand(command: WhatsAppCommand): void {\n const config = loadCommandsConfig();\n\n // Check if command already exists\n const existingIndex = config.commands.findIndex(\n (c) => c.name.toLowerCase() === command.name.toLowerCase()\n );\n\n if (existingIndex >= 0) {\n config.commands[existingIndex] = command;\n } else {\n config.commands.push(command);\n }\n\n saveCommandsConfig(config);\n}\n\n/**\n * Remove a custom command\n */\nexport function removeCommand(name: string): boolean {\n const config = loadCommandsConfig();\n const initialLength = config.commands.length;\n\n config.commands = config.commands.filter(\n (c) => c.name.toLowerCase() !== name.toLowerCase()\n );\n\n if (config.commands.length < initialLength) {\n saveCommandsConfig(config);\n return true;\n }\n\n return false;\n}\n\n/**\n * Get list of available commands\n */\nexport function getAvailableCommands(): WhatsAppCommand[] {\n const config = loadCommandsConfig();\n return config.commands.filter((c) => c.enabled);\n}\n"],
5
+ "mappings": "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",
6
+ "names": []
7
+ }