@askexenow/exe-os 0.8.83 → 0.8.85

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 (95) hide show
  1. package/dist/bin/backfill-conversations.js +746 -595
  2. package/dist/bin/backfill-responses.js +745 -594
  3. package/dist/bin/backfill-vectors.js +312 -226
  4. package/dist/bin/cleanup-stale-review-tasks.js +97 -2
  5. package/dist/bin/cli.js +14350 -12518
  6. package/dist/bin/exe-agent.js +97 -88
  7. package/dist/bin/exe-assign.js +1003 -854
  8. package/dist/bin/exe-boot.js +1257 -320
  9. package/dist/bin/exe-call.js +10 -0
  10. package/dist/bin/exe-cloud.js +29 -6
  11. package/dist/bin/exe-dispatch.js +210 -34
  12. package/dist/bin/exe-doctor.js +403 -6
  13. package/dist/bin/exe-export-behaviors.js +175 -72
  14. package/dist/bin/exe-forget.js +97 -2
  15. package/dist/bin/exe-gateway.js +550 -171
  16. package/dist/bin/exe-healthcheck.js +1 -0
  17. package/dist/bin/exe-heartbeat.js +100 -5
  18. package/dist/bin/exe-kill.js +175 -72
  19. package/dist/bin/exe-launch-agent.js +189 -76
  20. package/dist/bin/exe-link.js +902 -80
  21. package/dist/bin/exe-new-employee.js +38 -8
  22. package/dist/bin/exe-pending-messages.js +96 -2
  23. package/dist/bin/exe-pending-notifications.js +97 -2
  24. package/dist/bin/exe-pending-reviews.js +98 -3
  25. package/dist/bin/exe-rename.js +564 -23
  26. package/dist/bin/exe-review.js +231 -73
  27. package/dist/bin/exe-search.js +989 -226
  28. package/dist/bin/exe-session-cleanup.js +4806 -1665
  29. package/dist/bin/exe-settings.js +20 -5
  30. package/dist/bin/exe-status.js +97 -2
  31. package/dist/bin/exe-team.js +97 -2
  32. package/dist/bin/git-sweep.js +899 -207
  33. package/dist/bin/graph-backfill.js +175 -72
  34. package/dist/bin/graph-export.js +175 -72
  35. package/dist/bin/install.js +38 -7
  36. package/dist/bin/list-providers.js +1 -0
  37. package/dist/bin/scan-tasks.js +904 -211
  38. package/dist/bin/setup.js +867 -268
  39. package/dist/bin/shard-migrate.js +175 -72
  40. package/dist/bin/update.js +1 -0
  41. package/dist/bin/wiki-sync.js +175 -72
  42. package/dist/gateway/index.js +548 -166
  43. package/dist/hooks/bug-report-worker.js +208 -23
  44. package/dist/hooks/commit-complete.js +897 -205
  45. package/dist/hooks/error-recall.js +988 -226
  46. package/dist/hooks/ingest-worker.js +1638 -1194
  47. package/dist/hooks/ingest.js +3 -0
  48. package/dist/hooks/instructions-loaded.js +707 -97
  49. package/dist/hooks/notification.js +699 -89
  50. package/dist/hooks/post-compact.js +714 -104
  51. package/dist/hooks/pre-compact.js +897 -205
  52. package/dist/hooks/pre-tool-use.js +742 -123
  53. package/dist/hooks/prompt-ingest-worker.js +242 -101
  54. package/dist/hooks/prompt-submit.js +995 -233
  55. package/dist/hooks/response-ingest-worker.js +242 -101
  56. package/dist/hooks/session-end.js +3941 -400
  57. package/dist/hooks/session-start.js +1001 -226
  58. package/dist/hooks/stop.js +725 -115
  59. package/dist/hooks/subagent-stop.js +714 -104
  60. package/dist/hooks/summary-worker.js +1964 -1330
  61. package/dist/index.js +1651 -1053
  62. package/dist/lib/cloud-sync.js +907 -86
  63. package/dist/lib/consolidation.js +2 -1
  64. package/dist/lib/database.js +642 -87
  65. package/dist/lib/db-daemon-client.js +503 -0
  66. package/dist/lib/device-registry.js +547 -7
  67. package/dist/lib/embedder.js +14 -28
  68. package/dist/lib/employee-templates.js +84 -74
  69. package/dist/lib/employees.js +9 -0
  70. package/dist/lib/exe-daemon-client.js +16 -29
  71. package/dist/lib/exe-daemon.js +1955 -922
  72. package/dist/lib/hybrid-search.js +988 -226
  73. package/dist/lib/identity.js +87 -67
  74. package/dist/lib/keychain.js +9 -1
  75. package/dist/lib/messaging.js +8 -1
  76. package/dist/lib/reminders.js +91 -74
  77. package/dist/lib/schedules.js +96 -2
  78. package/dist/lib/skill-learning.js +103 -85
  79. package/dist/lib/store.js +234 -73
  80. package/dist/lib/tasks.js +111 -22
  81. package/dist/lib/tmux-routing.js +120 -31
  82. package/dist/lib/token-spend.js +273 -0
  83. package/dist/lib/ws-client.js +11 -0
  84. package/dist/mcp/server.js +5222 -475
  85. package/dist/mcp/tools/complete-reminder.js +94 -77
  86. package/dist/mcp/tools/create-reminder.js +94 -77
  87. package/dist/mcp/tools/create-task.js +120 -22
  88. package/dist/mcp/tools/deactivate-behavior.js +95 -77
  89. package/dist/mcp/tools/list-reminders.js +94 -77
  90. package/dist/mcp/tools/list-tasks.js +31 -1
  91. package/dist/mcp/tools/send-message.js +8 -1
  92. package/dist/mcp/tools/update-task.js +39 -10
  93. package/dist/runtime/index.js +911 -219
  94. package/dist/tui/App.js +997 -295
  95. package/package.json +6 -1
@@ -0,0 +1,273 @@
1
+ var __getOwnPropNames = Object.getOwnPropertyNames;
2
+ var __esm = (fn, res) => function __init() {
3
+ return fn && (res = (0, fn[__getOwnPropNames(fn)[0]])(fn = 0)), res;
4
+ };
5
+
6
+ // src/lib/config.ts
7
+ import { readFile, writeFile, mkdir, chmod } from "fs/promises";
8
+ import { readFileSync, existsSync, renameSync } from "fs";
9
+ import path from "path";
10
+ import os from "os";
11
+ function resolveDataDir() {
12
+ if (process.env.EXE_OS_DIR) return process.env.EXE_OS_DIR;
13
+ if (process.env.EXE_MEM_DIR) return process.env.EXE_MEM_DIR;
14
+ const newDir = path.join(os.homedir(), ".exe-os");
15
+ const legacyDir = path.join(os.homedir(), ".exe-mem");
16
+ if (!existsSync(newDir) && existsSync(legacyDir)) {
17
+ try {
18
+ renameSync(legacyDir, newDir);
19
+ process.stderr.write(`[exe-os] Migrated data directory: ~/.exe-mem \u2192 ~/.exe-os
20
+ `);
21
+ } catch {
22
+ return legacyDir;
23
+ }
24
+ }
25
+ return newDir;
26
+ }
27
+ var EXE_AI_DIR, DB_PATH, MODELS_DIR, CONFIG_PATH, LEGACY_LANCE_PATH, CURRENT_CONFIG_VERSION, DEFAULT_CONFIG;
28
+ var init_config = __esm({
29
+ "src/lib/config.ts"() {
30
+ "use strict";
31
+ EXE_AI_DIR = resolveDataDir();
32
+ DB_PATH = path.join(EXE_AI_DIR, "memories.db");
33
+ MODELS_DIR = path.join(EXE_AI_DIR, "models");
34
+ CONFIG_PATH = path.join(EXE_AI_DIR, "config.json");
35
+ LEGACY_LANCE_PATH = path.join(EXE_AI_DIR, "local.lance");
36
+ CURRENT_CONFIG_VERSION = 1;
37
+ DEFAULT_CONFIG = {
38
+ config_version: CURRENT_CONFIG_VERSION,
39
+ dbPath: DB_PATH,
40
+ modelFile: "jina-embeddings-v5-small-q4_k_m.gguf",
41
+ embeddingDim: 1024,
42
+ batchSize: 20,
43
+ flushIntervalMs: 1e4,
44
+ autoIngestion: true,
45
+ autoRetrieval: true,
46
+ searchMode: "hybrid",
47
+ hookSearchMode: "hybrid",
48
+ fileGrepEnabled: true,
49
+ splashEffect: true,
50
+ consolidationEnabled: true,
51
+ consolidationIntervalMs: 6 * 60 * 60 * 1e3,
52
+ consolidationModel: "claude-haiku-4-5-20251001",
53
+ consolidationMaxCallsPerRun: 20,
54
+ selfQueryRouter: true,
55
+ selfQueryModel: "claude-haiku-4-5-20251001",
56
+ rerankerEnabled: true,
57
+ scalingRoadmap: {
58
+ rerankerAutoTrigger: {
59
+ enabled: true,
60
+ broadQueryMinCardinality: 5e4,
61
+ fetchTopK: 150,
62
+ returnTopK: 5
63
+ }
64
+ },
65
+ graphRagEnabled: true,
66
+ wikiEnabled: false,
67
+ wikiUrl: "",
68
+ wikiApiKey: "",
69
+ wikiSyncIntervalMs: 30 * 60 * 1e3,
70
+ wikiWorkspaceMapping: {},
71
+ wikiAutoUpdate: true,
72
+ wikiAutoUpdateThreshold: 0.5,
73
+ wikiAutoUpdateCreateNew: true,
74
+ skillLearning: true,
75
+ skillThreshold: 3,
76
+ skillModel: "claude-haiku-4-5-20251001",
77
+ exeHeartbeat: {
78
+ enabled: true,
79
+ intervalSeconds: 60,
80
+ staleInProgressThresholdHours: 2
81
+ },
82
+ sessionLifecycle: {
83
+ idleKillEnabled: true,
84
+ idleKillTicksRequired: 3,
85
+ idleKillIntercomAckWindowMs: 1e4,
86
+ maxAutoInstances: 10
87
+ },
88
+ autoUpdate: {
89
+ checkOnBoot: true,
90
+ autoInstall: false,
91
+ checkIntervalMs: 24 * 60 * 60 * 1e3
92
+ }
93
+ };
94
+ }
95
+ });
96
+
97
+ // src/lib/token-spend.ts
98
+ import { readdir } from "fs/promises";
99
+ import { createReadStream } from "fs";
100
+ import { createInterface } from "readline";
101
+ import path3 from "path";
102
+ import os3 from "os";
103
+
104
+ // src/lib/database.ts
105
+ import { createClient } from "@libsql/client";
106
+
107
+ // src/lib/employees.ts
108
+ init_config();
109
+ import { readFile as readFile2, writeFile as writeFile2, mkdir as mkdir2 } from "fs/promises";
110
+ import { existsSync as existsSync2, symlinkSync, readlinkSync, readFileSync as readFileSync2, renameSync as renameSync2, unlinkSync, writeFileSync } from "fs";
111
+ import { execSync } from "child_process";
112
+ import path2 from "path";
113
+ import os2 from "os";
114
+ var EMPLOYEES_PATH = path2.join(EXE_AI_DIR, "exe-employees.json");
115
+
116
+ // src/lib/database.ts
117
+ var _resilientClient = null;
118
+ var _daemonClient = null;
119
+ function getClient() {
120
+ if (!_resilientClient) {
121
+ throw new Error("Database client not initialized. Call initDatabase() first.");
122
+ }
123
+ if (process.env.EXE_IS_DAEMON === "1") {
124
+ return _resilientClient;
125
+ }
126
+ if (_daemonClient && _daemonClient._isDaemonActive()) {
127
+ return _daemonClient;
128
+ }
129
+ return _resilientClient;
130
+ }
131
+
132
+ // src/lib/token-spend.ts
133
+ var MODEL_PRICING = {
134
+ // Opus 4.5+ ($5/$25 — Anthropic price drop from original Opus 4)
135
+ "claude-opus-4-7": { input: 5 / 1e6, output: 25 / 1e6, cacheRead: 0.5 / 1e6, cacheWrite: 6.25 / 1e6 },
136
+ "claude-opus-4-6": { input: 5 / 1e6, output: 25 / 1e6, cacheRead: 0.5 / 1e6, cacheWrite: 6.25 / 1e6 },
137
+ "claude-opus-4-5": { input: 5 / 1e6, output: 25 / 1e6, cacheRead: 0.5 / 1e6, cacheWrite: 6.25 / 1e6 },
138
+ // Opus 4.0/4.1 (legacy $15/$75)
139
+ "claude-opus-4-1": { input: 15 / 1e6, output: 75 / 1e6, cacheRead: 1.5 / 1e6, cacheWrite: 18.75 / 1e6 },
140
+ "claude-opus-4": { input: 15 / 1e6, output: 75 / 1e6, cacheRead: 1.5 / 1e6, cacheWrite: 18.75 / 1e6 },
141
+ // Sonnet 4.x
142
+ "claude-sonnet-4": { input: 3 / 1e6, output: 15 / 1e6, cacheRead: 0.3 / 1e6, cacheWrite: 3.75 / 1e6 },
143
+ // Sonnet 3.7/3.5
144
+ "claude-3-7-sonnet": { input: 3 / 1e6, output: 15 / 1e6, cacheRead: 0.3 / 1e6, cacheWrite: 3.75 / 1e6 },
145
+ "claude-3-5-sonnet": { input: 3 / 1e6, output: 15 / 1e6, cacheRead: 0.3 / 1e6, cacheWrite: 3.75 / 1e6 },
146
+ // Haiku 4.5
147
+ "claude-haiku-4-5": { input: 1 / 1e6, output: 5 / 1e6, cacheRead: 0.1 / 1e6, cacheWrite: 1.25 / 1e6 },
148
+ // Haiku 3.5
149
+ "claude-3-5-haiku": { input: 0.8 / 1e6, output: 4 / 1e6, cacheRead: 0.08 / 1e6, cacheWrite: 1 / 1e6 },
150
+ // Opus 3
151
+ "claude-3-opus": { input: 15 / 1e6, output: 75 / 1e6, cacheRead: 1.5 / 1e6, cacheWrite: 18.75 / 1e6 },
152
+ // Sonnet 3
153
+ "claude-3-sonnet": { input: 3 / 1e6, output: 15 / 1e6, cacheRead: 0.3 / 1e6, cacheWrite: 3.75 / 1e6 },
154
+ // Haiku 3
155
+ "claude-3-haiku": { input: 0.25 / 1e6, output: 1.25 / 1e6, cacheRead: 0.03 / 1e6, cacheWrite: 0.3 / 1e6 }
156
+ };
157
+ var DEFAULT_PRICING = MODEL_PRICING["claude-sonnet-4"];
158
+ function getPricing(model) {
159
+ if (MODEL_PRICING[model]) return MODEL_PRICING[model];
160
+ const stripped = model.replace(/-\d{8}$/, "");
161
+ if (MODEL_PRICING[stripped]) return MODEL_PRICING[stripped];
162
+ const sortedKeys = Object.keys(MODEL_PRICING).sort((a, b) => b.length - a.length);
163
+ for (const key of sortedKeys) {
164
+ if (model.includes(key)) return MODEL_PRICING[key];
165
+ }
166
+ return DEFAULT_PRICING;
167
+ }
168
+ async function getAgentSpend(period = "7d") {
169
+ const cutoff = periodToCutoff(period);
170
+ const client = getClient();
171
+ const result = await client.execute({
172
+ sql: `SELECT session_uuid, agent_id FROM session_agent_map WHERE started_at >= ?`,
173
+ args: [cutoff]
174
+ });
175
+ if (result.rows.length === 0) return [];
176
+ const sessionAgent = /* @__PURE__ */ new Map();
177
+ for (const row of result.rows) {
178
+ sessionAgent.set(row.session_uuid, row.agent_id);
179
+ }
180
+ const claudeDir = path3.join(os3.homedir(), ".claude", "projects");
181
+ let projectDirs = [];
182
+ try {
183
+ const entries = await readdir(claudeDir);
184
+ projectDirs = entries.map((e) => path3.join(claudeDir, e));
185
+ } catch {
186
+ return [];
187
+ }
188
+ const agentTotals = /* @__PURE__ */ new Map();
189
+ for (const [sessionUuid, agentId] of sessionAgent) {
190
+ for (const dir of projectDirs) {
191
+ const jsonlPath = path3.join(dir, `${sessionUuid}.jsonl`);
192
+ try {
193
+ const usage = await extractSessionUsage(jsonlPath);
194
+ if (usage.input === 0 && usage.output === 0) continue;
195
+ const totals = agentTotals.get(agentId) ?? {
196
+ input: 0,
197
+ output: 0,
198
+ cacheRead: 0,
199
+ cacheCreate: 0,
200
+ costUSD: 0,
201
+ sessions: /* @__PURE__ */ new Set()
202
+ };
203
+ totals.input += usage.input;
204
+ totals.output += usage.output;
205
+ totals.cacheRead += usage.cacheRead;
206
+ totals.cacheCreate += usage.cacheCreate;
207
+ totals.costUSD += usage.costUSD;
208
+ totals.sessions.add(sessionUuid);
209
+ agentTotals.set(agentId, totals);
210
+ break;
211
+ } catch {
212
+ }
213
+ }
214
+ }
215
+ return Array.from(agentTotals.entries()).map(([agentId, t]) => ({
216
+ agentId,
217
+ inputTokens: t.input,
218
+ outputTokens: t.output,
219
+ cacheReadTokens: t.cacheRead,
220
+ cacheCreationTokens: t.cacheCreate,
221
+ costUSD: t.costUSD,
222
+ sessions: t.sessions.size,
223
+ period
224
+ })).sort((a, b) => b.costUSD - a.costUSD);
225
+ }
226
+ async function extractSessionUsage(jsonlPath) {
227
+ let input = 0;
228
+ let output = 0;
229
+ let cacheRead = 0;
230
+ let cacheCreate = 0;
231
+ let costUSD = 0;
232
+ const seenMessageIds = /* @__PURE__ */ new Set();
233
+ const rl = createInterface({
234
+ input: createReadStream(jsonlPath, { encoding: "utf8" }),
235
+ crlfDelay: Infinity
236
+ });
237
+ for await (const line of rl) {
238
+ if (!line.includes('"type":"assistant"')) continue;
239
+ try {
240
+ const record = JSON.parse(line);
241
+ if (record.type !== "assistant") continue;
242
+ const messageId = record.message?.id;
243
+ if (messageId) {
244
+ if (seenMessageIds.has(messageId)) continue;
245
+ seenMessageIds.add(messageId);
246
+ }
247
+ const usage = record.message?.usage;
248
+ if (!usage) continue;
249
+ const model = record.message?.model ?? "";
250
+ const pricing = getPricing(model);
251
+ const inp = usage.input_tokens ?? 0;
252
+ const out = usage.output_tokens ?? 0;
253
+ const cr = usage.cache_read_input_tokens ?? 0;
254
+ const cc = usage.cache_creation_input_tokens ?? 0;
255
+ input += inp;
256
+ output += out;
257
+ cacheRead += cr;
258
+ cacheCreate += cc;
259
+ if (pricing) {
260
+ costUSD += inp * pricing.input + out * pricing.output + cr * pricing.cacheRead + cc * pricing.cacheWrite;
261
+ }
262
+ } catch {
263
+ }
264
+ }
265
+ return { input, output, cacheRead, cacheCreate, costUSD };
266
+ }
267
+ function periodToCutoff(period) {
268
+ const ms = { "24h": 864e5, "7d": 6048e5, "30d": 2592e6 }[period];
269
+ return new Date(Date.now() - ms).toISOString();
270
+ }
271
+ export {
272
+ getAgentSpend
273
+ };
@@ -25,12 +25,14 @@ function assertSecureWsUrl(url) {
25
25
  var MIN_RECONNECT_MS = 1e3;
26
26
  var MAX_RECONNECT_MS = 3e4;
27
27
  var HEARTBEAT_INTERVAL_MS = 3e4;
28
+ var MAX_RECONNECT_ATTEMPTS = 10;
28
29
  function createWsClient(config, handlers) {
29
30
  assertSecureWsUrl(config.endpoint);
30
31
  let ws = null;
31
32
  let connected = false;
32
33
  let closed = false;
33
34
  let reconnectDelay = MIN_RECONNECT_MS;
35
+ let reconnectAttempts = 0;
34
36
  let reconnectTimer = null;
35
37
  let heartbeatTimer = null;
36
38
  let currentAgents = [];
@@ -47,6 +49,7 @@ function createWsClient(config, handlers) {
47
49
  ws.onopen = () => {
48
50
  connected = true;
49
51
  reconnectDelay = MIN_RECONNECT_MS;
52
+ reconnectAttempts = 0;
50
53
  handlers.onConnect();
51
54
  sendRaw({
52
55
  type: "register",
@@ -113,6 +116,14 @@ function createWsClient(config, handlers) {
113
116
  }
114
117
  function scheduleReconnect() {
115
118
  if (closed || reconnectTimer) return;
119
+ reconnectAttempts++;
120
+ if (reconnectAttempts > MAX_RECONNECT_ATTEMPTS) {
121
+ process.stderr.write(
122
+ `[ws-client] Cloud sync not available \u2014 gave up after ${MAX_RECONNECT_ATTEMPTS} attempts. Check your network or endpoint config.
123
+ `
124
+ );
125
+ return;
126
+ }
116
127
  const jitter = reconnectDelay * (0.75 + Math.random() * 0.5);
117
128
  reconnectTimer = setTimeout(() => {
118
129
  reconnectTimer = null;