@agenticmail/enterprise 0.5.381 → 0.5.382

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 (74) hide show
  1. package/dist/dashboard/app.js +8 -4
  2. package/logs/cloudflared-error.log +14 -0
  3. package/package.json +1 -1
  4. package/dist/agent-heartbeat-3JIAQFMW.js +0 -510
  5. package/dist/agent-heartbeat-BICTJ4TT.js +0 -510
  6. package/dist/agent-heartbeat-HZCGY3KS.js +0 -510
  7. package/dist/agent-heartbeat-VU7RMEIG.js +0 -510
  8. package/dist/browser-tool-P57PLVW2.js +0 -4002
  9. package/dist/cli-agent-45NEZY7H.js +0 -2483
  10. package/dist/cli-agent-6UJP34H3.js +0 -2483
  11. package/dist/cli-agent-A4IFO5ML.js +0 -2483
  12. package/dist/cli-agent-CFLWIYRJ.js +0 -2483
  13. package/dist/cli-agent-H3DNHT44.js +0 -2483
  14. package/dist/cli-agent-HWRINZSE.js +0 -2483
  15. package/dist/cli-agent-NIKMPSIT.js +0 -2497
  16. package/dist/cli-agent-VEQZKUCM.js +0 -2483
  17. package/dist/cli-agent-XCILLLBL.js +0 -2483
  18. package/dist/cli-agent-ZZGZ5KDE.js +0 -2483
  19. package/dist/cli-recover-CO2RNTTL.js +0 -488
  20. package/dist/cli-serve-2K7AX2G2.js +0 -281
  21. package/dist/cli-serve-42HTQ45A.js +0 -281
  22. package/dist/cli-serve-5KMU2NHN.js +0 -281
  23. package/dist/cli-serve-7AL3NKYK.js +0 -281
  24. package/dist/cli-serve-AEW7G6YZ.js +0 -281
  25. package/dist/cli-serve-COPCKGDT.js +0 -281
  26. package/dist/cli-serve-H5LPD45H.js +0 -286
  27. package/dist/cli-serve-IOSXZHIK.js +0 -286
  28. package/dist/cli-serve-MGKI4Z5G.js +0 -281
  29. package/dist/cli-serve-WFOG5TRD.js +0 -281
  30. package/dist/cli-validate-ZO3PW64W.js +0 -150
  31. package/dist/cli-verify-7EMGBE46.js +0 -149
  32. package/dist/dynamodb-QS64UREL.js +0 -424
  33. package/dist/factory-RTZU2K54.js +0 -11
  34. package/dist/meetings-EMKU56G3.js +0 -12
  35. package/dist/mongodb-MWH3DGZY.js +0 -320
  36. package/dist/mysql-E54MUUH2.js +0 -580
  37. package/dist/postgres-KAERODLZ.js +0 -879
  38. package/dist/routes-NIQHEAV2.js +0 -92
  39. package/dist/routes-OKGFOUUP.js +0 -92
  40. package/dist/routes-Q3YMFXX3.js +0 -92
  41. package/dist/routes-XTMY7IYO.js +0 -92
  42. package/dist/runtime-F75SNKYK.js +0 -45
  43. package/dist/runtime-GS2SAORH.js +0 -45
  44. package/dist/runtime-HNPFFKQL.js +0 -45
  45. package/dist/runtime-NST6HJUG.js +0 -45
  46. package/dist/runtime-RDULR4UE.js +0 -45
  47. package/dist/runtime-SUDXHYMB.js +0 -45
  48. package/dist/runtime-UZEFUBAA.js +0 -45
  49. package/dist/runtime-WKWJ5BGT.js +0 -45
  50. package/dist/server-4XD4C4GA.js +0 -28
  51. package/dist/server-CQPAITXK.js +0 -28
  52. package/dist/server-EGWD64TG.js +0 -28
  53. package/dist/server-ET4HLIC7.js +0 -28
  54. package/dist/server-I533SQLB.js +0 -28
  55. package/dist/server-NENS3OG6.js +0 -28
  56. package/dist/server-OYPIKSG2.js +0 -28
  57. package/dist/server-Q52YXWZD.js +0 -28
  58. package/dist/server-RHXEW7LO.js +0 -28
  59. package/dist/server-XRW327IX.js +0 -28
  60. package/dist/setup-4DKCCU6S.js +0 -20
  61. package/dist/setup-C22ILWUJ.js +0 -20
  62. package/dist/setup-DCBFFTLT.js +0 -20
  63. package/dist/setup-DKOGTOIG.js +0 -20
  64. package/dist/setup-FDTGGTZE.js +0 -20
  65. package/dist/setup-JLD2NH4N.js +0 -20
  66. package/dist/setup-LQT3AYBX.js +0 -20
  67. package/dist/setup-QKWRY6L3.js +0 -20
  68. package/dist/setup-T2U77CEY.js +0 -20
  69. package/dist/setup-WO337O3I.js +0 -20
  70. package/dist/skills-4MHEPJGC.js +0 -16
  71. package/dist/sqlite-YZ64WWE6.js +0 -572
  72. package/dist/telegram-QRNGRT5M.js +0 -17
  73. package/dist/turso-H7HBWPHO.js +0 -501
  74. package/dist/whatsapp-32CNVRFX.js +0 -31
@@ -198,6 +198,10 @@ function App() {
198
198
  var s = await apiCall('/settings');
199
199
  if (s && s.name) setCompanyName(s.name);
200
200
  } catch {}
201
+ // Check for updates (admin/owner only)
202
+ if (u.role === 'admin' || u.role === 'owner') {
203
+ apiCall('/admin/system/update-check').then(ui => { if (ui?.updateAvailable) setUpdateInfo(ui); }).catch(() => {});
204
+ }
201
205
  setAuthed(true);
202
206
  setAuthChecked(true);
203
207
  }).catch(() => setAuthChecked(true));
@@ -334,10 +338,6 @@ function App() {
334
338
  if (d?.user) {
335
339
  setUser(d.user);
336
340
  if (!d.user.totpEnabled) setShow2faReminder(true);
337
- // Check for updates (admin only)
338
- if (d.user.role === 'admin' || d.user.role === 'owner') {
339
- apiCall('/admin/system/update-check').then(u => { if (u?.updateAvailable) setUpdateInfo(u); }).catch(() => {});
340
- }
341
341
  }
342
342
  if (d?.mustResetPassword) setMustResetPassword(true);
343
343
  // Init encryption before enabling dashboard
@@ -349,6 +349,10 @@ function App() {
349
349
  if (window.__transportEncryption) await window.__transportEncryption.waitForReady();
350
350
  }
351
351
  } catch {}
352
+ // Check for updates AFTER encryption is ready (admin only)
353
+ if (d?.user && (d.user.role === 'admin' || d.user.role === 'owner')) {
354
+ apiCall('/admin/system/update-check').then(u => { if (u?.updateAvailable) setUpdateInfo(u); }).catch(() => {});
355
+ }
352
356
  setAuthed(true);
353
357
  } });
354
358
 
@@ -0,0 +1,14 @@
1
+ 2026-03-06 01:32:21: 2026-03-06T00:32:21Z ERR error="stream 6461 canceled by remote with error code 0" connIndex=0 event=1 ingressRule=0 originService=http://localhost:3100
2
+ 2026-03-06 01:32:21: 2026-03-06T00:32:21Z ERR Request failed error="stream 6461 canceled by remote with error code 0" connIndex=0 dest=https://enterprise.agenticmail.io/api/engine/agent-status-stream event=0 ip=198.41.192.107 type=http
3
+ 2026-03-06 01:32:28: 2026-03-06T00:32:28Z ERR error="stream 6765 canceled by remote with error code 0" connIndex=0 event=1 ingressRule=0 originService=http://localhost:3100
4
+ 2026-03-06 01:32:28: 2026-03-06T00:32:28Z ERR Request failed error="stream 6765 canceled by remote with error code 0" connIndex=0 dest=https://enterprise.agenticmail.io/api/engine/agent-status-stream event=0 ip=198.41.192.107 type=http
5
+ 2026-03-06 01:35:51: 2026-03-06T00:35:51Z ERR error="stream 7769 canceled by remote with error code 0" connIndex=0 event=1 ingressRule=0 originService=http://localhost:3100
6
+ 2026-03-06 01:35:51: 2026-03-06T00:35:51Z ERR Request failed error="stream 7769 canceled by remote with error code 0" connIndex=0 dest=https://enterprise.agenticmail.io/api/engine/agent-status-stream event=0 ip=198.41.192.107 type=http
7
+ 2026-03-06 01:35:54: 2026-03-06T00:35:54Z ERR error="stream 7705 canceled by remote with error code 0" connIndex=0 event=1 ingressRule=0 originService=http://localhost:3100
8
+ 2026-03-06 01:35:54: 2026-03-06T00:35:54Z ERR Request failed error="stream 7705 canceled by remote with error code 0" connIndex=0 dest=https://enterprise.agenticmail.io/api/engine/agent-status-stream?agentId=3eecd57d-03ae-440d-8945-5b35f43a8d90 event=0 ip=198.41.192.107 type=http
9
+ 2026-03-06 01:35:56: 2026-03-06T00:35:56Z ERR error="stream 7737 canceled by remote with error code 0" connIndex=0 event=1 ingressRule=0 originService=http://localhost:3100
10
+ 2026-03-06 01:35:56: 2026-03-06T00:35:56Z ERR Request failed error="stream 7737 canceled by remote with error code 0" connIndex=0 dest=https://enterprise.agenticmail.io/api/engine/agent-status-stream?agentId=3eecd57d-03ae-440d-8945-5b35f43a8d90 event=0 ip=198.41.192.107 type=http
11
+ 2026-03-06 01:37:51: 2026-03-06T00:37:51Z ERR error="stream 8069 canceled by remote with error code 0" connIndex=0 event=1 ingressRule=0 originService=http://localhost:3100
12
+ 2026-03-06 01:37:51: 2026-03-06T00:37:51Z ERR Request failed error="stream 8069 canceled by remote with error code 0" connIndex=0 dest=https://enterprise.agenticmail.io/api/engine/agent-status-stream event=0 ip=198.41.192.107 type=http
13
+ 2026-03-06 01:37:58: 2026-03-06T00:37:58Z ERR error="stream 8361 canceled by remote with error code 0" connIndex=0 event=1 ingressRule=0 originService=http://localhost:3100
14
+ 2026-03-06 01:37:58: 2026-03-06T00:37:58Z ERR Request failed error="stream 8361 canceled by remote with error code 0" connIndex=0 dest=https://enterprise.agenticmail.io/api/engine/agent-status-stream event=0 ip=198.41.192.107 type=http
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@agenticmail/enterprise",
3
- "version": "0.5.381",
3
+ "version": "0.5.382",
4
4
  "description": "AgenticMail Enterprise — cloud-hosted AI agent identity, email, auth & compliance for organizations",
5
5
  "type": "module",
6
6
  "bin": {
@@ -1,510 +0,0 @@
1
- import "./chunk-KFQGP6VL.js";
2
-
3
- // src/engine/agent-heartbeat.ts
4
- var DEFAULT_SETTINGS = {
5
- enabled: true,
6
- baseIntervalMs: 5 * 6e4,
7
- // 5 minutes
8
- maxIntervalMs: 30 * 6e4,
9
- // 30 minutes
10
- dampingFactor: 0.5,
11
- maxBatchSize: 5,
12
- quietHoursStart: 23,
13
- quietHoursEnd: 8
14
- };
15
- function createUnreadEmailCheck() {
16
- return {
17
- id: "unread_emails",
18
- name: "Unread Emails",
19
- intervalMs: 10 * 6e4,
20
- // 10 minutes
21
- priority: "high",
22
- requiresClockIn: false,
23
- // emails can be urgent outside hours
24
- consecutiveNoOps: 0,
25
- enabled: true,
26
- check: async (ctx) => {
27
- try {
28
- const rows = await ctx.db.query(
29
- `SELECT COUNT(*) as cnt FROM agent_memory
30
- WHERE agent_id = $1
31
- AND category = 'processed_email'
32
- AND created_at > NOW() - INTERVAL '4 hours'`,
33
- [ctx.agentId]
34
- );
35
- const processedRecently = parseInt(rows?.[0]?.cnt || "0");
36
- const lastProcessed = await ctx.db.query(
37
- `SELECT MAX(created_at) as last_at FROM agent_memory
38
- WHERE agent_id = $1 AND category = 'processed_email'`,
39
- [ctx.agentId]
40
- );
41
- const lastAt = lastProcessed?.[0]?.last_at;
42
- if (ctx.isWorkHours && processedRecently === 0 && ctx.hour >= 10) {
43
- return {
44
- needsAction: true,
45
- summary: "No emails processed in the last 4 hours during work hours. Check inbox.",
46
- priority: "medium",
47
- data: { lastProcessedAt: lastAt }
48
- };
49
- }
50
- return { needsAction: false, priority: "low" };
51
- } catch {
52
- return { needsAction: false, priority: "low" };
53
- }
54
- }
55
- };
56
- }
57
- function createUpcomingEventsCheck() {
58
- return {
59
- id: "upcoming_events",
60
- name: "Upcoming Calendar Events",
61
- intervalMs: 15 * 6e4,
62
- // 15 minutes
63
- priority: "high",
64
- requiresClockIn: false,
65
- consecutiveNoOps: 0,
66
- enabled: true,
67
- check: async (ctx) => {
68
- try {
69
- const _twoHoursFromNow = new Date(ctx.now.getTime() + 2 * 60 * 60 * 1e3).toISOString();
70
- const rows = await ctx.db.query(
71
- `SELECT content FROM agent_memory
72
- WHERE agent_id = $1
73
- AND category = 'context'
74
- AND content LIKE '%meeting%'
75
- AND created_at > NOW() - INTERVAL '24 hours'
76
- ORDER BY created_at DESC LIMIT 5`,
77
- [ctx.agentId]
78
- );
79
- const upcomingMeetings = (rows || []).filter((r) => {
80
- const content = r.content || "";
81
- return content.includes("meeting") || content.includes("event");
82
- });
83
- if (upcomingMeetings.length > 0) {
84
- return {
85
- needsAction: true,
86
- summary: `${upcomingMeetings.length} potential upcoming event(s). Agent should check Google Calendar.`,
87
- priority: "high",
88
- data: { count: upcomingMeetings.length }
89
- };
90
- }
91
- return { needsAction: false, priority: "low" };
92
- } catch {
93
- return { needsAction: false, priority: "low" };
94
- }
95
- }
96
- };
97
- }
98
- function createStaleSessionsCheck() {
99
- return {
100
- id: "stale_sessions",
101
- name: "Stale Sessions",
102
- intervalMs: 30 * 6e4,
103
- // 30 minutes
104
- priority: "medium",
105
- requiresClockIn: true,
106
- consecutiveNoOps: 0,
107
- enabled: true,
108
- check: async (ctx) => {
109
- try {
110
- const rows = await ctx.db.query(
111
- `SELECT COUNT(*) as cnt FROM agent_sessions
112
- WHERE agent_id = $1
113
- AND status = 'active'
114
- AND updated_at < NOW() - INTERVAL '2 hours'`,
115
- [ctx.agentId]
116
- );
117
- const staleCount = parseInt(rows?.[0]?.cnt || "0");
118
- if (staleCount > 0) {
119
- return {
120
- needsAction: true,
121
- summary: `${staleCount} stale session(s) detected (active > 2 hours with no updates). May need cleanup.`,
122
- priority: "medium",
123
- data: { staleCount }
124
- };
125
- }
126
- return { needsAction: false, priority: "low" };
127
- } catch {
128
- return { needsAction: false, priority: "low" };
129
- }
130
- }
131
- };
132
- }
133
- function createMemoryHealthCheck() {
134
- return {
135
- id: "memory_health",
136
- name: "Memory Health",
137
- intervalMs: 60 * 6e4,
138
- // 1 hour
139
- priority: "low",
140
- requiresClockIn: true,
141
- consecutiveNoOps: 0,
142
- enabled: true,
143
- check: async (ctx) => {
144
- try {
145
- const rows = await ctx.db.query(
146
- `SELECT COUNT(*) as cnt FROM agent_memory
147
- WHERE agent_id = $1 AND created_at > NOW() - INTERVAL '24 hours'`,
148
- [ctx.agentId]
149
- );
150
- const recentCount = parseInt(rows?.[0]?.cnt || "0");
151
- if (recentCount > 200) {
152
- return {
153
- needsAction: true,
154
- summary: `Memory flood detected: ${recentCount} memories in 24h (threshold: 200). Consider pruning.`,
155
- priority: "medium",
156
- data: { recentCount }
157
- };
158
- }
159
- return { needsAction: false, priority: "low" };
160
- } catch {
161
- return { needsAction: false, priority: "low" };
162
- }
163
- }
164
- };
165
- }
166
- function createUnansweredChatCheck() {
167
- return {
168
- id: "unanswered_chat",
169
- name: "Unanswered Chat Messages",
170
- intervalMs: 5 * 6e4,
171
- // 5 minutes
172
- priority: "high",
173
- requiresClockIn: false,
174
- consecutiveNoOps: 0,
175
- enabled: true,
176
- check: async (ctx) => {
177
- try {
178
- const rows = await ctx.db.query(
179
- `SELECT COUNT(*) as cnt FROM agent_sessions
180
- WHERE agent_id = $1
181
- AND status = 'failed'
182
- AND metadata::text LIKE '%chat%'
183
- AND created_at > NOW() - INTERVAL '1 hour'`,
184
- [ctx.agentId]
185
- );
186
- const failedChats = parseInt(rows?.[0]?.cnt || "0");
187
- if (failedChats > 0) {
188
- return {
189
- needsAction: true,
190
- summary: `${failedChats} failed chat session(s) in the last hour. Messages may be unanswered.`,
191
- priority: "urgent",
192
- data: { failedChats }
193
- };
194
- }
195
- return { needsAction: false, priority: "low" };
196
- } catch {
197
- return { needsAction: false, priority: "low" };
198
- }
199
- }
200
- };
201
- }
202
- function createTaskDeadlineCheck() {
203
- return {
204
- id: "task_deadlines",
205
- name: "Task Deadlines",
206
- intervalMs: 60 * 6e4,
207
- // 1 hour
208
- priority: "medium",
209
- requiresClockIn: true,
210
- consecutiveNoOps: 0,
211
- enabled: true,
212
- check: async (ctx) => {
213
- try {
214
- const rows = await ctx.db.query(
215
- `SELECT COUNT(*) as cnt FROM agent_memory
216
- WHERE agent_id = $1
217
- AND category = 'context'
218
- AND importance = 'high'
219
- AND content LIKE '%deadline%'
220
- AND created_at > NOW() - INTERVAL '48 hours'`,
221
- [ctx.agentId]
222
- );
223
- const urgentTasks = parseInt(rows?.[0]?.cnt || "0");
224
- if (urgentTasks > 0) {
225
- return {
226
- needsAction: true,
227
- summary: `${urgentTasks} task(s) with approaching deadlines. Agent should review and prioritize.`,
228
- priority: "high",
229
- data: { urgentTasks }
230
- };
231
- }
232
- return { needsAction: false, priority: "low" };
233
- } catch {
234
- return { needsAction: false, priority: "low" };
235
- }
236
- }
237
- };
238
- }
239
- function createErrorRateCheck() {
240
- return {
241
- id: "error_rate",
242
- name: "Error Rate Monitor",
243
- intervalMs: 15 * 6e4,
244
- // 15 minutes
245
- priority: "high",
246
- requiresClockIn: false,
247
- // errors can happen anytime
248
- consecutiveNoOps: 0,
249
- enabled: true,
250
- check: async (ctx) => {
251
- try {
252
- const totalRows = await ctx.db.query(
253
- `SELECT COUNT(*) as cnt FROM agent_sessions
254
- WHERE agent_id = $1 AND created_at > NOW() - INTERVAL '1 hour'`,
255
- [ctx.agentId]
256
- );
257
- const failedRows = await ctx.db.query(
258
- `SELECT COUNT(*) as cnt FROM agent_sessions
259
- WHERE agent_id = $1 AND status = 'failed' AND created_at > NOW() - INTERVAL '1 hour'`,
260
- [ctx.agentId]
261
- );
262
- const total = parseInt(totalRows?.[0]?.cnt || "0");
263
- const failed = parseInt(failedRows?.[0]?.cnt || "0");
264
- if (total >= 3 && failed / total > 0.5) {
265
- return {
266
- needsAction: true,
267
- summary: `High error rate: ${failed}/${total} sessions failed in the last hour (${Math.round(failed / total * 100)}%). Possible infrastructure issue.`,
268
- priority: "urgent",
269
- data: { total, failed, rate: failed / total }
270
- };
271
- }
272
- return { needsAction: false, priority: "low" };
273
- } catch {
274
- return { needsAction: false, priority: "low" };
275
- }
276
- }
277
- };
278
- }
279
- var AgentHeartbeatManager = class {
280
- config;
281
- settings;
282
- checks = /* @__PURE__ */ new Map();
283
- tickTimer = null;
284
- globalConsecutiveNoOps = 0;
285
- lastActionTimestamp = 0;
286
- stats = {
287
- totalTicks: 0,
288
- totalChecksRun: 0,
289
- totalActionsTriggered: 0,
290
- totalTokensSaved: 0,
291
- // estimated tokens NOT spent due to no-op ticks
292
- startedAt: Date.now()
293
- };
294
- constructor(config, settings) {
295
- this.config = config;
296
- this.settings = { ...DEFAULT_SETTINGS, ...settings };
297
- this.registerCheck(createUnreadEmailCheck());
298
- this.registerCheck(createUpcomingEventsCheck());
299
- this.registerCheck(createStaleSessionsCheck());
300
- this.registerCheck(createMemoryHealthCheck());
301
- this.registerCheck(createUnansweredChatCheck());
302
- this.registerCheck(createTaskDeadlineCheck());
303
- this.registerCheck(createErrorRateCheck());
304
- if (config.enabledChecks) {
305
- for (const [id, enabled] of Object.entries(config.enabledChecks)) {
306
- const check = this.checks.get(id);
307
- if (check) check.enabled = enabled;
308
- }
309
- }
310
- }
311
- registerCheck(check) {
312
- this.checks.set(check.id, check);
313
- }
314
- async start() {
315
- if (!this.settings.enabled) {
316
- console.log("[heartbeat] Disabled, skipping");
317
- return;
318
- }
319
- console.log(`[heartbeat] Starting with ${this.checks.size} checks, base interval ${this.settings.baseIntervalMs / 1e3}s`);
320
- setTimeout(() => this.tick(), 6e4);
321
- this.scheduleNextTick();
322
- }
323
- stop() {
324
- if (this.tickTimer) {
325
- clearTimeout(this.tickTimer);
326
- this.tickTimer = null;
327
- }
328
- console.log(`[heartbeat] Stopped. Stats: ${this.stats.totalTicks} ticks, ${this.stats.totalChecksRun} checks, ${this.stats.totalActionsTriggered} actions, ~${this.stats.totalTokensSaved} tokens saved`);
329
- }
330
- getStats() {
331
- return {
332
- ...this.stats,
333
- uptimeMs: Date.now() - this.stats.startedAt,
334
- currentIntervalMs: this.calculateInterval(),
335
- globalConsecutiveNoOps: this.globalConsecutiveNoOps,
336
- checks: Array.from(this.checks.values()).map((c) => ({
337
- id: c.id,
338
- name: c.name,
339
- enabled: c.enabled,
340
- lastRunAt: c.lastRunAt,
341
- lastActionAt: c.lastActionAt,
342
- consecutiveNoOps: c.consecutiveNoOps
343
- }))
344
- };
345
- }
346
- // ─── Core Tick Logic ────────────────────────────────
347
- async tick() {
348
- this.stats.totalTicks++;
349
- const now = /* @__PURE__ */ new Date();
350
- const tz = this.config.timezone || "UTC";
351
- const localTime = new Date(now.toLocaleString("en-US", { timeZone: tz }));
352
- const hour = localTime.getHours();
353
- const minute = localTime.getMinutes();
354
- const dayOfWeek = localTime.getDay();
355
- const isWorkHours = this.isWithinWorkHours(hour, minute, dayOfWeek);
356
- const isQuietHours = this.isQuietHours(hour);
357
- const isClockedIn = this.config.isClockedIn();
358
- const ctx = {
359
- agentId: this.config.agentId,
360
- orgId: this.config.orgId,
361
- agentName: this.config.agentName,
362
- role: this.config.role,
363
- managerEmail: this.config.managerEmail,
364
- timezone: tz,
365
- db: this.config.db,
366
- now,
367
- localTime,
368
- hour,
369
- minute,
370
- dayOfWeek,
371
- isWorkHours,
372
- isClockedIn
373
- };
374
- const actionableItems = [];
375
- for (const check of this.checks.values()) {
376
- if (!check.enabled) continue;
377
- if (check.requiresClockIn && !isClockedIn) continue;
378
- if (isQuietHours && check.priority !== "high") continue;
379
- const adaptiveInterval = Math.min(
380
- check.intervalMs * (1 + this.settings.dampingFactor * check.consecutiveNoOps),
381
- this.settings.maxIntervalMs
382
- );
383
- const timeSinceLastRun = now.getTime() - (check.lastRunAt || 0);
384
- if (timeSinceLastRun < adaptiveInterval) continue;
385
- this.stats.totalChecksRun++;
386
- check.lastRunAt = now.getTime();
387
- try {
388
- const result = await check.check(ctx);
389
- if (result.needsAction) {
390
- actionableItems.push({ check, result });
391
- check.consecutiveNoOps = 0;
392
- check.lastActionAt = now.getTime();
393
- } else {
394
- check.consecutiveNoOps++;
395
- this.stats.totalTokensSaved += 500;
396
- }
397
- } catch (err) {
398
- console.warn(`[heartbeat] Check ${check.id} error: ${err.message}`);
399
- check.consecutiveNoOps++;
400
- }
401
- }
402
- if (actionableItems.length > 0) {
403
- this.globalConsecutiveNoOps = 0;
404
- this.lastActionTimestamp = now.getTime();
405
- this.stats.totalActionsTriggered++;
406
- const priorityOrder = { urgent: 0, high: 1, medium: 2, low: 3 };
407
- actionableItems.sort((a, b) => priorityOrder[a.result.priority] - priorityOrder[b.result.priority]);
408
- const batch = actionableItems.slice(0, this.settings.maxBatchSize);
409
- await this.dispatchBatch(batch, ctx);
410
- } else {
411
- this.globalConsecutiveNoOps++;
412
- }
413
- this.scheduleNextTick();
414
- }
415
- /**
416
- * Calculate next tick interval using damped oscillator model:
417
- * interval = base * (1 + damping * consecutiveNoOps)
418
- *
419
- * With fibonacci-like acceleration:
420
- * After 0 no-ops: 5min (base)
421
- * After 1 no-op: 7.5min
422
- * After 2 no-ops: 10min
423
- * After 4 no-ops: 15min
424
- * After 8 no-ops: 25min
425
- * After 10 no-ops: 30min (max)
426
- *
427
- * Resets to base immediately when action is detected.
428
- */
429
- calculateInterval() {
430
- const base = this.settings.baseIntervalMs;
431
- const max = this.settings.maxIntervalMs;
432
- const damping = this.settings.dampingFactor;
433
- const noOps = this.globalConsecutiveNoOps;
434
- return Math.min(base * (1 + damping * noOps), max);
435
- }
436
- scheduleNextTick() {
437
- if (this.tickTimer) clearTimeout(this.tickTimer);
438
- const interval = this.calculateInterval();
439
- this.tickTimer = setTimeout(() => this.tick(), interval);
440
- this.tickTimer.unref();
441
- }
442
- isWithinWorkHours(hour, minute, dayOfWeek) {
443
- const schedule = this.config.schedule;
444
- if (!schedule) return true;
445
- const isWorkday = schedule.days.includes(dayOfWeek);
446
- const timeStr = `${String(hour).padStart(2, "0")}:${String(minute).padStart(2, "0")}`;
447
- const isWithinHours = timeStr >= schedule.start && timeStr < schedule.end;
448
- return isWorkday && isWithinHours;
449
- }
450
- isQuietHours(hour) {
451
- const start = this.settings.quietHoursStart;
452
- const end = this.settings.quietHoursEnd;
453
- if (start === void 0 || end === void 0) return false;
454
- if (start > end) {
455
- return hour >= start || hour < end;
456
- }
457
- return hour >= start && hour < end;
458
- }
459
- /**
460
- * Dispatch a batch of actionable items to the agent via one LLM session.
461
- * This is the ONLY place tokens are spent.
462
- */
463
- async dispatchBatch(items, ctx) {
464
- if (!this.config.runtime) {
465
- console.warn("[heartbeat] No runtime \u2014 cannot dispatch actions");
466
- return;
467
- }
468
- try {
469
- const { guardrails } = await import("./routes-NIQHEAV2.js");
470
- const status = await guardrails.getStatus(ctx.agentId);
471
- if (status.paused || status.offDuty) {
472
- console.log(`[heartbeat] Skipping action dispatch \u2014 agent is ${status.offDuty ? "off duty" : "paused"}`);
473
- return;
474
- }
475
- } catch {
476
- }
477
- const summaries = items.map(
478
- (item, i) => `${i + 1}. [${item.result.priority.toUpperCase()}] ${item.check.name}: ${item.result.summary}`
479
- ).join("\n");
480
- const prompt = `HEARTBEAT ALERT \u2014 The following ${items.length} item(s) need your attention:
481
-
482
- ${summaries}
483
-
484
- For each item, take the appropriate action:
485
- - For unread emails: Check your inbox with gmail_search and respond to any urgent ones.
486
- - For upcoming events: Check google_calendar_list for the next 2 hours and prepare.
487
- - For stale sessions: Review and close any stuck sessions.
488
- - For unanswered chats: Check Google Chat and respond.
489
- - For error rate issues: Investigate recent failures and notify your manager if critical.
490
- - For task deadlines: Review google_tasks_list and prioritize.
491
- - For memory health: Consider pruning old or low-importance memories.
492
-
493
- Be efficient \u2014 handle what you can and note what needs human intervention.
494
- If something needs your manager's attention, email ${ctx.managerEmail || "your manager"}.`;
495
- const systemPrompt = `You are ${ctx.agentName}, a ${ctx.role}. This is an automated heartbeat check \u2014 items flagged as needing attention. Handle them efficiently. Don't create unnecessary work \u2014 only act on what's genuinely important.`;
496
- try {
497
- const session = await this.config.runtime.spawnSession({
498
- agentId: ctx.agentId,
499
- message: prompt,
500
- systemPrompt
501
- });
502
- console.log(`[heartbeat] \u2705 Action session ${session.id} dispatched (${items.length} items: ${items.map((i) => i.check.id).join(", ")})`);
503
- } catch (err) {
504
- console.error(`[heartbeat] Failed to dispatch action: ${err.message}`);
505
- }
506
- }
507
- };
508
- export {
509
- AgentHeartbeatManager
510
- };