@stackmemoryai/stackmemory 0.5.31 → 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 (63) hide show
  1. package/dist/cli/claude-sm.js +199 -16
  2. package/dist/cli/claude-sm.js.map +2 -2
  3. package/dist/cli/commands/context.js +0 -11
  4. package/dist/cli/commands/context.js.map +2 -2
  5. package/dist/cli/commands/linear.js +1 -14
  6. package/dist/cli/commands/linear.js.map +2 -2
  7. package/dist/cli/commands/login.js +32 -10
  8. package/dist/cli/commands/login.js.map +2 -2
  9. package/dist/cli/commands/migrate.js +80 -22
  10. package/dist/cli/commands/migrate.js.map +2 -2
  11. package/dist/cli/commands/model.js +533 -0
  12. package/dist/cli/commands/model.js.map +7 -0
  13. package/dist/cli/commands/ralph.js +93 -28
  14. package/dist/cli/commands/ralph.js.map +2 -2
  15. package/dist/cli/commands/service.js +10 -3
  16. package/dist/cli/commands/service.js.map +2 -2
  17. package/dist/cli/commands/skills.js +60 -10
  18. package/dist/cli/commands/skills.js.map +2 -2
  19. package/dist/cli/commands/sms-notify.js +342 -22
  20. package/dist/cli/commands/sms-notify.js.map +3 -3
  21. package/dist/cli/index.js +2 -0
  22. package/dist/cli/index.js.map +2 -2
  23. package/dist/core/context/dual-stack-manager.js +23 -7
  24. package/dist/core/context/dual-stack-manager.js.map +2 -2
  25. package/dist/core/context/frame-database.js +33 -5
  26. package/dist/core/context/frame-database.js.map +2 -2
  27. package/dist/core/context/frame-digest.js +6 -1
  28. package/dist/core/context/frame-digest.js.map +2 -2
  29. package/dist/core/context/frame-manager.js +56 -9
  30. package/dist/core/context/frame-manager.js.map +2 -2
  31. package/dist/core/context/permission-manager.js +0 -11
  32. package/dist/core/context/permission-manager.js.map +2 -2
  33. package/dist/core/context/recursive-context-manager.js +15 -9
  34. package/dist/core/context/recursive-context-manager.js.map +2 -2
  35. package/dist/core/context/shared-context-layer.js +0 -11
  36. package/dist/core/context/shared-context-layer.js.map +2 -2
  37. package/dist/core/context/validation.js +6 -1
  38. package/dist/core/context/validation.js.map +2 -2
  39. package/dist/core/models/fallback-monitor.js +229 -0
  40. package/dist/core/models/fallback-monitor.js.map +7 -0
  41. package/dist/core/models/model-router.js +331 -0
  42. package/dist/core/models/model-router.js.map +7 -0
  43. package/dist/hooks/claude-code-whatsapp-hook.js +197 -0
  44. package/dist/hooks/claude-code-whatsapp-hook.js.map +7 -0
  45. package/dist/hooks/linear-task-picker.js +1 -1
  46. package/dist/hooks/linear-task-picker.js.map +2 -2
  47. package/dist/hooks/schemas.js +55 -1
  48. package/dist/hooks/schemas.js.map +2 -2
  49. package/dist/hooks/session-summary.js +5 -1
  50. package/dist/hooks/session-summary.js.map +2 -2
  51. package/dist/hooks/sms-action-runner.js +12 -1
  52. package/dist/hooks/sms-action-runner.js.map +2 -2
  53. package/dist/hooks/sms-notify.js +4 -2
  54. package/dist/hooks/sms-notify.js.map +2 -2
  55. package/dist/hooks/sms-webhook.js +23 -2
  56. package/dist/hooks/sms-webhook.js.map +2 -2
  57. package/dist/hooks/whatsapp-commands.js +376 -0
  58. package/dist/hooks/whatsapp-commands.js.map +7 -0
  59. package/dist/hooks/whatsapp-scheduler.js +317 -0
  60. package/dist/hooks/whatsapp-scheduler.js.map +7 -0
  61. package/dist/hooks/whatsapp-sync.js +375 -0
  62. package/dist/hooks/whatsapp-sync.js.map +7 -0
  63. package/package.json +2 -3
@@ -0,0 +1,317 @@
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 { randomBytes } from "crypto";
9
+ import { writeFileSecure, ensureSecureDir } from "./secure-fs.js";
10
+ import { ScheduleStorageSchema, parseConfigSafe } from "./schemas.js";
11
+ import {
12
+ getFrameDigestData,
13
+ generateMobileDigest,
14
+ loadSyncOptions
15
+ } from "./whatsapp-sync.js";
16
+ import { sendNotification, loadSMSConfig } from "./sms-notify.js";
17
+ const STORAGE_PATH = join(homedir(), ".stackmemory", "whatsapp-schedules.json");
18
+ const DEFAULT_STORAGE = {
19
+ schedules: [],
20
+ lastChecked: (/* @__PURE__ */ new Date()).toISOString()
21
+ };
22
+ let schedulerInterval = null;
23
+ function loadStorage() {
24
+ try {
25
+ if (existsSync(STORAGE_PATH)) {
26
+ const data = JSON.parse(readFileSync(STORAGE_PATH, "utf8"));
27
+ return parseConfigSafe(
28
+ ScheduleStorageSchema,
29
+ data,
30
+ DEFAULT_STORAGE,
31
+ "whatsapp-schedules"
32
+ );
33
+ }
34
+ } catch {
35
+ }
36
+ return { ...DEFAULT_STORAGE, lastChecked: (/* @__PURE__ */ new Date()).toISOString() };
37
+ }
38
+ function saveStorage(storage) {
39
+ try {
40
+ ensureSecureDir(join(homedir(), ".stackmemory"));
41
+ writeFileSecure(STORAGE_PATH, JSON.stringify(storage, null, 2));
42
+ } catch {
43
+ }
44
+ }
45
+ function generateScheduleId() {
46
+ return randomBytes(6).toString("hex");
47
+ }
48
+ function parseTime(time) {
49
+ const match = time.match(/^(\d{2}):(\d{2})$/);
50
+ if (!match) return null;
51
+ const hours = parseInt(match[1], 10);
52
+ const minutes = parseInt(match[2], 10);
53
+ if (hours < 0 || hours > 23 || minutes < 0 || minutes > 59) {
54
+ return null;
55
+ }
56
+ return { hours, minutes };
57
+ }
58
+ function calculateNextRun(config, fromDate) {
59
+ const now = fromDate || /* @__PURE__ */ new Date();
60
+ switch (config.type) {
61
+ case "daily": {
62
+ const time = config.time ? parseTime(config.time) : { hours: 9, minutes: 0 };
63
+ if (!time) {
64
+ throw new Error(`Invalid time format: ${config.time}`);
65
+ }
66
+ const next = new Date(now);
67
+ next.setHours(time.hours, time.minutes, 0, 0);
68
+ if (next <= now) {
69
+ next.setDate(next.getDate() + 1);
70
+ }
71
+ return next;
72
+ }
73
+ case "hourly": {
74
+ const next = new Date(now);
75
+ next.setMinutes(0, 0, 0);
76
+ next.setHours(next.getHours() + 1);
77
+ return next;
78
+ }
79
+ case "interval": {
80
+ const intervalMinutes = config.intervalMinutes || 60;
81
+ const next = new Date(now.getTime() + intervalMinutes * 60 * 1e3);
82
+ return next;
83
+ }
84
+ default:
85
+ throw new Error(`Unknown schedule type: ${config.type}`);
86
+ }
87
+ }
88
+ function isQuietHours() {
89
+ const smsConfig = loadSMSConfig();
90
+ if (!smsConfig.quietHours?.enabled) {
91
+ return false;
92
+ }
93
+ const now = /* @__PURE__ */ new Date();
94
+ const currentMinutes = now.getHours() * 60 + now.getMinutes();
95
+ const startTime = parseTime(smsConfig.quietHours.start);
96
+ const endTime = parseTime(smsConfig.quietHours.end);
97
+ if (!startTime || !endTime) {
98
+ return false;
99
+ }
100
+ const startMinutes = startTime.hours * 60 + startTime.minutes;
101
+ const endMinutes = endTime.hours * 60 + endTime.minutes;
102
+ if (startMinutes > endMinutes) {
103
+ return currentMinutes >= startMinutes || currentMinutes < endMinutes;
104
+ }
105
+ return currentMinutes >= startMinutes && currentMinutes < endMinutes;
106
+ }
107
+ function scheduleDigest(config) {
108
+ const storage = loadStorage();
109
+ const id = generateScheduleId();
110
+ const nextRun = calculateNextRun(config);
111
+ const schedule = {
112
+ id,
113
+ config,
114
+ enabled: true,
115
+ nextRun: nextRun.toISOString(),
116
+ createdAt: (/* @__PURE__ */ new Date()).toISOString()
117
+ };
118
+ storage.schedules.push(schedule);
119
+ saveStorage(storage);
120
+ console.log(
121
+ `[whatsapp-scheduler] Created schedule ${id}, next run: ${nextRun.toISOString()}`
122
+ );
123
+ return id;
124
+ }
125
+ function cancelSchedule(scheduleId) {
126
+ const storage = loadStorage();
127
+ const initialLength = storage.schedules.length;
128
+ storage.schedules = storage.schedules.filter((s) => s.id !== scheduleId);
129
+ if (storage.schedules.length < initialLength) {
130
+ saveStorage(storage);
131
+ console.log(`[whatsapp-scheduler] Cancelled schedule ${scheduleId}`);
132
+ return true;
133
+ }
134
+ return false;
135
+ }
136
+ function setScheduleEnabled(scheduleId, enabled) {
137
+ const storage = loadStorage();
138
+ const schedule = storage.schedules.find((s) => s.id === scheduleId);
139
+ if (!schedule) {
140
+ return false;
141
+ }
142
+ schedule.enabled = enabled;
143
+ if (enabled) {
144
+ schedule.nextRun = calculateNextRun(schedule.config).toISOString();
145
+ }
146
+ saveStorage(storage);
147
+ return true;
148
+ }
149
+ function listSchedules() {
150
+ const storage = loadStorage();
151
+ return storage.schedules;
152
+ }
153
+ function getSchedule(scheduleId) {
154
+ const storage = loadStorage();
155
+ return storage.schedules.find((s) => s.id === scheduleId);
156
+ }
157
+ async function generateActivitySummary() {
158
+ const data = await getFrameDigestData();
159
+ if (!data) {
160
+ return null;
161
+ }
162
+ const options = loadSyncOptions();
163
+ return generateMobileDigest(data, options);
164
+ }
165
+ async function runScheduledDigest(scheduleId) {
166
+ const storage = loadStorage();
167
+ const schedule = storage.schedules.find((s) => s.id === scheduleId);
168
+ if (!schedule) {
169
+ return {
170
+ success: false,
171
+ sent: false,
172
+ error: `Schedule not found: ${scheduleId}`
173
+ };
174
+ }
175
+ if (!schedule.enabled) {
176
+ return { success: false, sent: false, error: "Schedule is disabled" };
177
+ }
178
+ if (schedule.config.quietHoursRespect && isQuietHours()) {
179
+ return {
180
+ success: true,
181
+ sent: false,
182
+ message: "Skipped due to quiet hours"
183
+ };
184
+ }
185
+ const digest = await generateActivitySummary();
186
+ if (!digest && !schedule.config.includeInactive) {
187
+ schedule.lastRun = (/* @__PURE__ */ new Date()).toISOString();
188
+ schedule.nextRun = calculateNextRun(schedule.config).toISOString();
189
+ saveStorage(storage);
190
+ return { success: true, sent: false, message: "No activity to report" };
191
+ }
192
+ const message = digest || "No recent activity. All systems idle.";
193
+ const result = await sendNotification({
194
+ type: "custom",
195
+ title: "Scheduled Digest",
196
+ message,
197
+ prompt: {
198
+ type: "options",
199
+ options: [
200
+ { key: "1", label: "Details", action: "stackmemory status" },
201
+ { key: "2", label: "Tasks", action: "stackmemory task list" }
202
+ ]
203
+ }
204
+ });
205
+ schedule.lastRun = (/* @__PURE__ */ new Date()).toISOString();
206
+ schedule.nextRun = calculateNextRun(schedule.config).toISOString();
207
+ saveStorage(storage);
208
+ if (result.success) {
209
+ return {
210
+ success: true,
211
+ sent: true,
212
+ message: `Digest sent (${message.length} chars)`
213
+ };
214
+ } else {
215
+ return { success: false, sent: false, error: result.error };
216
+ }
217
+ }
218
+ async function checkAndRunDueSchedules() {
219
+ const storage = loadStorage();
220
+ const now = /* @__PURE__ */ new Date();
221
+ let ran = 0;
222
+ let errors = 0;
223
+ for (const schedule of storage.schedules) {
224
+ if (!schedule.enabled || !schedule.nextRun) {
225
+ continue;
226
+ }
227
+ const nextRun = new Date(schedule.nextRun);
228
+ if (nextRun <= now) {
229
+ console.log(`[whatsapp-scheduler] Running due schedule ${schedule.id}`);
230
+ const result = await runScheduledDigest(schedule.id);
231
+ if (result.success) {
232
+ if (result.sent) {
233
+ ran++;
234
+ }
235
+ } else {
236
+ errors++;
237
+ console.error(
238
+ `[whatsapp-scheduler] Schedule ${schedule.id} failed: ${result.error}`
239
+ );
240
+ }
241
+ }
242
+ }
243
+ storage.lastChecked = now.toISOString();
244
+ saveStorage(storage);
245
+ return { checked: storage.schedules.length, ran, errors };
246
+ }
247
+ function startScheduler(checkIntervalMs = 6e4) {
248
+ if (schedulerInterval) {
249
+ console.log("[whatsapp-scheduler] Scheduler already running");
250
+ return;
251
+ }
252
+ console.log(
253
+ `[whatsapp-scheduler] Starting scheduler (interval: ${checkIntervalMs}ms)`
254
+ );
255
+ checkAndRunDueSchedules().catch(console.error);
256
+ schedulerInterval = setInterval(() => {
257
+ checkAndRunDueSchedules().catch(console.error);
258
+ }, checkIntervalMs);
259
+ }
260
+ function stopScheduler() {
261
+ if (schedulerInterval) {
262
+ clearInterval(schedulerInterval);
263
+ schedulerInterval = null;
264
+ console.log("[whatsapp-scheduler] Scheduler stopped");
265
+ }
266
+ }
267
+ function isSchedulerRunning() {
268
+ return schedulerInterval !== null;
269
+ }
270
+ function scheduleDailyDigest(time) {
271
+ const parsed = parseTime(time);
272
+ if (!parsed) {
273
+ throw new Error(
274
+ `Invalid time format: ${time}. Use HH:MM format (e.g., 09:00)`
275
+ );
276
+ }
277
+ return scheduleDigest({
278
+ type: "daily",
279
+ time,
280
+ includeInactive: false,
281
+ quietHoursRespect: true
282
+ });
283
+ }
284
+ function scheduleHourlyDigest() {
285
+ return scheduleDigest({
286
+ type: "hourly",
287
+ includeInactive: false,
288
+ quietHoursRespect: true
289
+ });
290
+ }
291
+ function scheduleIntervalDigest(intervalMinutes) {
292
+ if (intervalMinutes < 5 || intervalMinutes > 1440) {
293
+ throw new Error("Interval must be between 5 and 1440 minutes");
294
+ }
295
+ return scheduleDigest({
296
+ type: "interval",
297
+ intervalMinutes,
298
+ includeInactive: false,
299
+ quietHoursRespect: true
300
+ });
301
+ }
302
+ export {
303
+ cancelSchedule,
304
+ checkAndRunDueSchedules,
305
+ getSchedule,
306
+ isSchedulerRunning,
307
+ listSchedules,
308
+ runScheduledDigest,
309
+ scheduleDailyDigest,
310
+ scheduleDigest,
311
+ scheduleHourlyDigest,
312
+ scheduleIntervalDigest,
313
+ setScheduleEnabled,
314
+ startScheduler,
315
+ stopScheduler
316
+ };
317
+ //# sourceMappingURL=whatsapp-scheduler.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/hooks/whatsapp-scheduler.ts"],
4
+ "sourcesContent": ["/**\n * WhatsApp Scheduled Digest Manager\n * Schedule periodic context digests\n */\n\nimport { existsSync, readFileSync } from 'fs';\nimport { join } from 'path';\nimport { homedir } from 'os';\nimport { randomBytes } from 'crypto';\nimport { writeFileSecure, ensureSecureDir } from './secure-fs.js';\nimport { ScheduleStorageSchema, parseConfigSafe } from './schemas.js';\nimport {\n getFrameDigestData,\n generateMobileDigest,\n loadSyncOptions,\n} from './whatsapp-sync.js';\nimport { sendNotification, loadSMSConfig } from './sms-notify.js';\n\nexport interface ScheduleConfig {\n type: 'daily' | 'hourly' | 'interval';\n time?: string; // \"HH:MM\" for daily\n intervalMinutes?: number; // for interval type\n includeInactive?: boolean; // include when no activity\n quietHoursRespect: boolean;\n}\n\nexport interface Schedule {\n id: string;\n config: ScheduleConfig;\n enabled: boolean;\n lastRun?: string;\n nextRun?: string;\n createdAt: string;\n}\n\ninterface ScheduleStorage {\n schedules: Schedule[];\n lastChecked: string;\n}\n\nconst STORAGE_PATH = join(homedir(), '.stackmemory', 'whatsapp-schedules.json');\n\nconst DEFAULT_STORAGE: ScheduleStorage = {\n schedules: [],\n lastChecked: new Date().toISOString(),\n};\n\n// Active scheduler interval handle\nlet schedulerInterval: NodeJS.Timeout | null = null;\n\n/**\n * Load schedule storage\n */\nfunction loadStorage(): ScheduleStorage {\n try {\n if (existsSync(STORAGE_PATH)) {\n const data = JSON.parse(readFileSync(STORAGE_PATH, 'utf8'));\n return parseConfigSafe(\n ScheduleStorageSchema,\n data,\n DEFAULT_STORAGE,\n 'whatsapp-schedules'\n );\n }\n } catch {\n // Use defaults\n }\n return { ...DEFAULT_STORAGE, lastChecked: new Date().toISOString() };\n}\n\n/**\n * Save schedule storage\n */\nfunction saveStorage(storage: ScheduleStorage): void {\n try {\n ensureSecureDir(join(homedir(), '.stackmemory'));\n writeFileSecure(STORAGE_PATH, JSON.stringify(storage, null, 2));\n } catch {\n // Silently fail\n }\n}\n\n/**\n * Generate unique schedule ID\n */\nfunction generateScheduleId(): string {\n return randomBytes(6).toString('hex');\n}\n\n/**\n * Parse time string to hours and minutes\n */\nfunction parseTime(time: string): { hours: number; minutes: number } | null {\n const match = time.match(/^(\\d{2}):(\\d{2})$/);\n if (!match) return null;\n\n const hours = parseInt(match[1], 10);\n const minutes = parseInt(match[2], 10);\n\n if (hours < 0 || hours > 23 || minutes < 0 || minutes > 59) {\n return null;\n }\n\n return { hours, minutes };\n}\n\n/**\n * Calculate next run time for a schedule\n */\nfunction calculateNextRun(config: ScheduleConfig, fromDate?: Date): Date {\n const now = fromDate || new Date();\n\n switch (config.type) {\n case 'daily': {\n const time = config.time\n ? parseTime(config.time)\n : { hours: 9, minutes: 0 };\n if (!time) {\n throw new Error(`Invalid time format: ${config.time}`);\n }\n\n const next = new Date(now);\n next.setHours(time.hours, time.minutes, 0, 0);\n\n // If the time has passed today, schedule for tomorrow\n if (next <= now) {\n next.setDate(next.getDate() + 1);\n }\n\n return next;\n }\n\n case 'hourly': {\n const next = new Date(now);\n next.setMinutes(0, 0, 0);\n next.setHours(next.getHours() + 1);\n return next;\n }\n\n case 'interval': {\n const intervalMinutes = config.intervalMinutes || 60;\n const next = new Date(now.getTime() + intervalMinutes * 60 * 1000);\n return next;\n }\n\n default:\n throw new Error(`Unknown schedule type: ${config.type}`);\n }\n}\n\n/**\n * Check if current time is within quiet hours\n */\nfunction isQuietHours(): boolean {\n const smsConfig = loadSMSConfig();\n\n if (!smsConfig.quietHours?.enabled) {\n return false;\n }\n\n const now = new Date();\n const currentMinutes = now.getHours() * 60 + now.getMinutes();\n\n const startTime = parseTime(smsConfig.quietHours.start);\n const endTime = parseTime(smsConfig.quietHours.end);\n\n if (!startTime || !endTime) {\n return false;\n }\n\n const startMinutes = startTime.hours * 60 + startTime.minutes;\n const endMinutes = endTime.hours * 60 + endTime.minutes;\n\n // Handle overnight quiet hours (e.g., 22:00 - 08:00)\n if (startMinutes > endMinutes) {\n return currentMinutes >= startMinutes || currentMinutes < endMinutes;\n }\n\n return currentMinutes >= startMinutes && currentMinutes < endMinutes;\n}\n\n/**\n * Schedule a periodic digest\n */\nexport function scheduleDigest(config: ScheduleConfig): string {\n const storage = loadStorage();\n\n const id = generateScheduleId();\n const nextRun = calculateNextRun(config);\n\n const schedule: Schedule = {\n id,\n config,\n enabled: true,\n nextRun: nextRun.toISOString(),\n createdAt: new Date().toISOString(),\n };\n\n storage.schedules.push(schedule);\n saveStorage(storage);\n\n console.log(\n `[whatsapp-scheduler] Created schedule ${id}, next run: ${nextRun.toISOString()}`\n );\n\n return id;\n}\n\n/**\n * Cancel a schedule\n */\nexport function cancelSchedule(scheduleId: string): boolean {\n const storage = loadStorage();\n const initialLength = storage.schedules.length;\n\n storage.schedules = storage.schedules.filter((s) => s.id !== scheduleId);\n\n if (storage.schedules.length < initialLength) {\n saveStorage(storage);\n console.log(`[whatsapp-scheduler] Cancelled schedule ${scheduleId}`);\n return true;\n }\n\n return false;\n}\n\n/**\n * Enable/disable a schedule\n */\nexport function setScheduleEnabled(\n scheduleId: string,\n enabled: boolean\n): boolean {\n const storage = loadStorage();\n const schedule = storage.schedules.find((s) => s.id === scheduleId);\n\n if (!schedule) {\n return false;\n }\n\n schedule.enabled = enabled;\n\n if (enabled) {\n schedule.nextRun = calculateNextRun(schedule.config).toISOString();\n }\n\n saveStorage(storage);\n return true;\n}\n\n/**\n * List all schedules\n */\nexport function listSchedules(): Schedule[] {\n const storage = loadStorage();\n return storage.schedules;\n}\n\n/**\n * Get a specific schedule\n */\nexport function getSchedule(scheduleId: string): Schedule | undefined {\n const storage = loadStorage();\n return storage.schedules.find((s) => s.id === scheduleId);\n}\n\n/**\n * Generate activity summary for digest\n */\nasync function generateActivitySummary(): Promise<string | null> {\n const data = await getFrameDigestData();\n\n if (!data) {\n return null;\n }\n\n const options = loadSyncOptions();\n return generateMobileDigest(data, options);\n}\n\n/**\n * Run a scheduled digest now\n */\nexport async function runScheduledDigest(scheduleId: string): Promise<{\n success: boolean;\n sent: boolean;\n message?: string;\n error?: string;\n}> {\n const storage = loadStorage();\n const schedule = storage.schedules.find((s) => s.id === scheduleId);\n\n if (!schedule) {\n return {\n success: false,\n sent: false,\n error: `Schedule not found: ${scheduleId}`,\n };\n }\n\n if (!schedule.enabled) {\n return { success: false, sent: false, error: 'Schedule is disabled' };\n }\n\n // Check quiet hours\n if (schedule.config.quietHoursRespect && isQuietHours()) {\n return {\n success: true,\n sent: false,\n message: 'Skipped due to quiet hours',\n };\n }\n\n // Generate digest\n const digest = await generateActivitySummary();\n\n if (!digest && !schedule.config.includeInactive) {\n // Update next run time even if we don't send\n schedule.lastRun = new Date().toISOString();\n schedule.nextRun = calculateNextRun(schedule.config).toISOString();\n saveStorage(storage);\n\n return { success: true, sent: false, message: 'No activity to report' };\n }\n\n const message = digest || 'No recent activity. All systems idle.';\n\n // Send notification\n const result = await sendNotification({\n type: 'custom',\n title: 'Scheduled Digest',\n message,\n prompt: {\n type: 'options',\n options: [\n { key: '1', label: 'Details', action: 'stackmemory status' },\n { key: '2', label: 'Tasks', action: 'stackmemory task list' },\n ],\n },\n });\n\n // Update schedule\n schedule.lastRun = new Date().toISOString();\n schedule.nextRun = calculateNextRun(schedule.config).toISOString();\n saveStorage(storage);\n\n if (result.success) {\n return {\n success: true,\n sent: true,\n message: `Digest sent (${message.length} chars)`,\n };\n } else {\n return { success: false, sent: false, error: result.error };\n }\n}\n\n/**\n * Check and run due schedules\n */\nexport async function checkAndRunDueSchedules(): Promise<{\n checked: number;\n ran: number;\n errors: number;\n}> {\n const storage = loadStorage();\n const now = new Date();\n\n let ran = 0;\n let errors = 0;\n\n for (const schedule of storage.schedules) {\n if (!schedule.enabled || !schedule.nextRun) {\n continue;\n }\n\n const nextRun = new Date(schedule.nextRun);\n\n if (nextRun <= now) {\n console.log(`[whatsapp-scheduler] Running due schedule ${schedule.id}`);\n\n const result = await runScheduledDigest(schedule.id);\n\n if (result.success) {\n if (result.sent) {\n ran++;\n }\n } else {\n errors++;\n console.error(\n `[whatsapp-scheduler] Schedule ${schedule.id} failed: ${result.error}`\n );\n }\n }\n }\n\n // Update last checked\n storage.lastChecked = now.toISOString();\n saveStorage(storage);\n\n return { checked: storage.schedules.length, ran, errors };\n}\n\n/**\n * Start the scheduler daemon\n */\nexport function startScheduler(checkIntervalMs: number = 60000): void {\n if (schedulerInterval) {\n console.log('[whatsapp-scheduler] Scheduler already running');\n return;\n }\n\n console.log(\n `[whatsapp-scheduler] Starting scheduler (interval: ${checkIntervalMs}ms)`\n );\n\n // Run immediately\n checkAndRunDueSchedules().catch(console.error);\n\n // Then run on interval\n schedulerInterval = setInterval(() => {\n checkAndRunDueSchedules().catch(console.error);\n }, checkIntervalMs);\n}\n\n/**\n * Stop the scheduler daemon\n */\nexport function stopScheduler(): void {\n if (schedulerInterval) {\n clearInterval(schedulerInterval);\n schedulerInterval = null;\n console.log('[whatsapp-scheduler] Scheduler stopped');\n }\n}\n\n/**\n * Check if scheduler is running\n */\nexport function isSchedulerRunning(): boolean {\n return schedulerInterval !== null;\n}\n\n/**\n * Create a daily schedule at specified time\n */\nexport function scheduleDailyDigest(time: string): string {\n const parsed = parseTime(time);\n if (!parsed) {\n throw new Error(\n `Invalid time format: ${time}. Use HH:MM format (e.g., 09:00)`\n );\n }\n\n return scheduleDigest({\n type: 'daily',\n time,\n includeInactive: false,\n quietHoursRespect: true,\n });\n}\n\n/**\n * Create an hourly schedule\n */\nexport function scheduleHourlyDigest(): string {\n return scheduleDigest({\n type: 'hourly',\n includeInactive: false,\n quietHoursRespect: true,\n });\n}\n\n/**\n * Create an interval-based schedule\n */\nexport function scheduleIntervalDigest(intervalMinutes: number): string {\n if (intervalMinutes < 5 || intervalMinutes > 1440) {\n throw new Error('Interval must be between 5 and 1440 minutes');\n }\n\n return scheduleDigest({\n type: 'interval',\n intervalMinutes,\n includeInactive: false,\n quietHoursRespect: true,\n });\n}\n"],
5
+ "mappings": "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",
6
+ "names": []
7
+ }