@askexenow/exe-os 0.9.61 → 0.9.63

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 (103) hide show
  1. package/deploy/stack-manifests/v0.9.json +62 -13
  2. package/dist/bin/backfill-conversations.js +210 -2
  3. package/dist/bin/backfill-responses.js +210 -2
  4. package/dist/bin/backfill-vectors.js +47 -2
  5. package/dist/bin/cleanup-stale-review-tasks.js +210 -2
  6. package/dist/bin/cli.js +488 -25
  7. package/dist/bin/exe-agent-config.js +2 -2
  8. package/dist/bin/exe-agent.js +2 -2
  9. package/dist/bin/exe-assign.js +210 -2
  10. package/dist/bin/exe-boot.js +47 -2
  11. package/dist/bin/exe-call.js +2 -2
  12. package/dist/bin/exe-cloud.js +2 -2
  13. package/dist/bin/exe-dispatch.js +210 -2
  14. package/dist/bin/exe-doctor.js +2691 -1212
  15. package/dist/bin/exe-export-behaviors.js +210 -2
  16. package/dist/bin/exe-forget.js +264 -2
  17. package/dist/bin/exe-gateway.js +210 -2
  18. package/dist/bin/exe-heartbeat.js +212 -4
  19. package/dist/bin/exe-kill.js +210 -2
  20. package/dist/bin/exe-launch-agent.js +210 -2
  21. package/dist/bin/exe-link.js +47 -2
  22. package/dist/bin/exe-new-employee.js +113 -3
  23. package/dist/bin/exe-pending-messages.js +210 -2
  24. package/dist/bin/exe-pending-notifications.js +210 -2
  25. package/dist/bin/exe-pending-reviews.js +210 -2
  26. package/dist/bin/exe-rename.js +210 -2
  27. package/dist/bin/exe-review.js +210 -2
  28. package/dist/bin/exe-search.js +234 -3
  29. package/dist/bin/exe-session-cleanup.js +210 -2
  30. package/dist/bin/exe-settings.js +2 -2
  31. package/dist/bin/exe-start-codex.js +322 -4
  32. package/dist/bin/exe-start-opencode.js +336 -4
  33. package/dist/bin/exe-status.js +210 -2
  34. package/dist/bin/exe-team.js +210 -2
  35. package/dist/bin/git-sweep.js +210 -2
  36. package/dist/bin/graph-backfill.js +210 -2
  37. package/dist/bin/graph-export.js +210 -2
  38. package/dist/bin/install.js +113 -3
  39. package/dist/bin/intercom-check.js +210 -2
  40. package/dist/bin/scan-tasks.js +210 -2
  41. package/dist/bin/setup.js +47 -2
  42. package/dist/bin/shard-migrate.js +210 -2
  43. package/dist/bin/stack-update.js +158 -13
  44. package/dist/bin/update.js +2 -2
  45. package/dist/gateway/index.js +210 -2
  46. package/dist/hooks/bug-report-worker.js +321 -44
  47. package/dist/hooks/codex-stop-task-finalizer.js +210 -2
  48. package/dist/hooks/commit-complete.js +210 -2
  49. package/dist/hooks/error-recall.js +234 -3
  50. package/dist/hooks/exe-heartbeat-hook.js +2 -2
  51. package/dist/hooks/ingest-worker.js +2 -2
  52. package/dist/hooks/ingest.js +210 -2
  53. package/dist/hooks/instructions-loaded.js +210 -2
  54. package/dist/hooks/notification.js +210 -2
  55. package/dist/hooks/post-compact.js +210 -2
  56. package/dist/hooks/post-tool-combined.js +234 -3
  57. package/dist/hooks/pre-compact.js +446 -111
  58. package/dist/hooks/pre-tool-use.js +210 -2
  59. package/dist/hooks/prompt-submit.js +234 -3
  60. package/dist/hooks/session-end.js +347 -124
  61. package/dist/hooks/session-start.js +236 -5
  62. package/dist/hooks/stop.js +210 -2
  63. package/dist/hooks/subagent-stop.js +210 -2
  64. package/dist/hooks/summary-worker.js +171 -74
  65. package/dist/index.js +210 -2
  66. package/dist/lib/agent-config.js +2 -2
  67. package/dist/lib/cloud-sync.js +47 -2
  68. package/dist/lib/config.js +2 -2
  69. package/dist/lib/consolidation.js +2 -2
  70. package/dist/lib/database.js +47 -2
  71. package/dist/lib/db-daemon-client.js +2 -2
  72. package/dist/lib/db.js +47 -2
  73. package/dist/lib/device-registry.js +47 -2
  74. package/dist/lib/embedder.js +2 -2
  75. package/dist/lib/employee-templates.js +2 -2
  76. package/dist/lib/employees.js +2 -2
  77. package/dist/lib/exe-daemon-client.js +2 -2
  78. package/dist/lib/exe-daemon.js +592 -10
  79. package/dist/lib/hybrid-search.js +234 -3
  80. package/dist/lib/identity.js +2 -2
  81. package/dist/lib/license.js +2 -2
  82. package/dist/lib/messaging.js +2 -2
  83. package/dist/lib/reminders.js +2 -2
  84. package/dist/lib/schedules.js +47 -2
  85. package/dist/lib/skill-learning.js +2 -2
  86. package/dist/lib/store.js +210 -2
  87. package/dist/lib/task-router.js +2 -2
  88. package/dist/lib/tasks.js +2 -2
  89. package/dist/lib/tmux-routing.js +2 -2
  90. package/dist/lib/token-spend.js +2 -2
  91. package/dist/mcp/server.js +608 -10
  92. package/dist/mcp/tools/complete-reminder.js +2 -2
  93. package/dist/mcp/tools/create-reminder.js +2 -2
  94. package/dist/mcp/tools/create-task.js +2 -2
  95. package/dist/mcp/tools/deactivate-behavior.js +2 -2
  96. package/dist/mcp/tools/list-reminders.js +2 -2
  97. package/dist/mcp/tools/list-tasks.js +2 -2
  98. package/dist/mcp/tools/send-message.js +2 -2
  99. package/dist/mcp/tools/update-task.js +2 -2
  100. package/dist/runtime/index.js +210 -2
  101. package/dist/tui/App.js +210 -2
  102. package/package.json +3 -2
  103. package/stack.release.json +23 -7
@@ -258,8 +258,8 @@ var init_config = __esm({
258
258
  rerankerAutoTrigger: {
259
259
  enabled: true,
260
260
  broadQueryMinCardinality: 5e4,
261
- fetchTopK: 150,
262
- returnTopK: 5
261
+ fetchTopK: 200,
262
+ returnTopK: 20
263
263
  }
264
264
  },
265
265
  graphRagEnabled: true,
@@ -2732,6 +2732,51 @@ async function ensureSchema() {
2732
2732
  VALUES (new.rowid, new.content_text, new.sender_name, new.agent_response);
2733
2733
  END;
2734
2734
  `);
2735
+ await client.executeMultiple(`
2736
+ CREATE TABLE IF NOT EXISTS memory_cards (
2737
+ id TEXT PRIMARY KEY,
2738
+ memory_id TEXT NOT NULL,
2739
+ agent_id TEXT NOT NULL,
2740
+ session_id TEXT NOT NULL,
2741
+ project_name TEXT,
2742
+ timestamp TEXT NOT NULL,
2743
+ card_type TEXT NOT NULL,
2744
+ subject TEXT,
2745
+ predicate TEXT,
2746
+ object TEXT,
2747
+ content TEXT NOT NULL,
2748
+ source_ref TEXT,
2749
+ confidence REAL DEFAULT 0.6,
2750
+ active INTEGER DEFAULT 1,
2751
+ created_at TEXT NOT NULL
2752
+ );
2753
+
2754
+ CREATE INDEX IF NOT EXISTS idx_memory_cards_agent
2755
+ ON memory_cards(agent_id, active, timestamp);
2756
+
2757
+ CREATE INDEX IF NOT EXISTS idx_memory_cards_memory
2758
+ ON memory_cards(memory_id);
2759
+
2760
+ CREATE VIRTUAL TABLE IF NOT EXISTS memory_cards_fts
2761
+ USING fts5(content, subject, predicate, object, content='memory_cards', content_rowid='rowid');
2762
+
2763
+ CREATE TRIGGER IF NOT EXISTS memory_cards_fts_ai AFTER INSERT ON memory_cards BEGIN
2764
+ INSERT INTO memory_cards_fts(rowid, content, subject, predicate, object)
2765
+ VALUES (new.rowid, new.content, new.subject, new.predicate, new.object);
2766
+ END;
2767
+
2768
+ CREATE TRIGGER IF NOT EXISTS memory_cards_fts_ad AFTER DELETE ON memory_cards BEGIN
2769
+ INSERT INTO memory_cards_fts(memory_cards_fts, rowid, content, subject, predicate, object)
2770
+ VALUES('delete', old.rowid, old.content, old.subject, old.predicate, old.object);
2771
+ END;
2772
+
2773
+ CREATE TRIGGER IF NOT EXISTS memory_cards_fts_au AFTER UPDATE ON memory_cards BEGIN
2774
+ INSERT INTO memory_cards_fts(memory_cards_fts, rowid, content, subject, predicate, object)
2775
+ VALUES('delete', old.rowid, old.content, old.subject, old.predicate, old.object);
2776
+ INSERT INTO memory_cards_fts(rowid, content, subject, predicate, object)
2777
+ VALUES (new.rowid, new.content, new.subject, new.predicate, new.object);
2778
+ END;
2779
+ `);
2735
2780
  try {
2736
2781
  await client.execute({
2737
2782
  sql: `ALTER TABLE memories ADD COLUMN tier INTEGER DEFAULT 3`,
@@ -4021,6 +4066,164 @@ ${p.content}`).join("\n\n");
4021
4066
  }
4022
4067
  });
4023
4068
 
4069
+ // src/lib/memory-cards.ts
4070
+ var memory_cards_exports = {};
4071
+ __export(memory_cards_exports, {
4072
+ extractMemoryCards: () => extractMemoryCards,
4073
+ insertMemoryCardsForBatch: () => insertMemoryCardsForBatch,
4074
+ searchMemoryCards: () => searchMemoryCards
4075
+ });
4076
+ import { createHash as createHash2 } from "crypto";
4077
+ function stableId(memoryId, type, content) {
4078
+ return createHash2("sha256").update(`${memoryId}:${type}:${content}`).digest("hex").slice(0, 32);
4079
+ }
4080
+ function cleanText(text) {
4081
+ return text.replace(/```[\s\S]*?```/g, " ").replace(/<[^>]+>/g, " ").replace(/\s+/g, " ").trim();
4082
+ }
4083
+ function splitSentences(text) {
4084
+ return cleanText(text).split(/(?<=[.!?])\s+|\n+/).map((s) => s.trim()).filter((s) => s.length >= 24 && s.length <= MAX_SENTENCE_CHARS);
4085
+ }
4086
+ function inferCardType(sentence, toolName) {
4087
+ const lower = sentence.toLowerCase();
4088
+ if (toolName === "store_decision" || /\b(decided|decision|adr|approved|rejected)\b/.test(lower)) return "decision";
4089
+ if (/\b(prefers|preference|likes|dislikes|wants|doesn't want|does not want)\b/.test(lower)) return "preference";
4090
+ if (/\b(changed|updated|replaced|now|no longer|instead|supersedes)\b/.test(lower)) return "belief_update";
4091
+ if (toolName && ["Read", "Write", "Edit", "Bash"].includes(toolName)) return "code";
4092
+ if (/\b(meeting|deadline|shipped|launched|completed|failed|blocked|assigned|created)\b/.test(lower)) return "event";
4093
+ return "fact";
4094
+ }
4095
+ function extractSubject(sentence, agentId) {
4096
+ const explicit = sentence.match(/\b([A-Z][a-zA-Z0-9_-]{2,}(?:\s+[A-Z][a-zA-Z0-9_-]{2,})?)\b/);
4097
+ return explicit?.[1] ?? agentId;
4098
+ }
4099
+ function predicateFor(type) {
4100
+ switch (type) {
4101
+ case "preference":
4102
+ return "prefers";
4103
+ case "belief_update":
4104
+ return "updated";
4105
+ case "decision":
4106
+ return "decided";
4107
+ case "event":
4108
+ return "happened";
4109
+ case "code":
4110
+ return "implemented";
4111
+ default:
4112
+ return "states";
4113
+ }
4114
+ }
4115
+ function extractMemoryCards(row) {
4116
+ const sentences = splitSentences(row.raw_text);
4117
+ const cards = [];
4118
+ for (const sentence of sentences) {
4119
+ const type = inferCardType(sentence, row.tool_name);
4120
+ const subject = extractSubject(sentence, row.agent_id);
4121
+ const content = sentence.length > MAX_SENTENCE_CHARS ? `${sentence.slice(0, MAX_SENTENCE_CHARS - 1)}\u2026` : sentence;
4122
+ cards.push({
4123
+ id: stableId(row.id, type, content),
4124
+ memory_id: row.id,
4125
+ agent_id: row.agent_id,
4126
+ session_id: row.session_id,
4127
+ project_name: row.project_name ?? null,
4128
+ timestamp: row.timestamp,
4129
+ card_type: type,
4130
+ subject,
4131
+ predicate: predicateFor(type),
4132
+ object: content,
4133
+ content,
4134
+ source_ref: row.id,
4135
+ confidence: type === "fact" ? 0.55 : 0.65
4136
+ });
4137
+ if (cards.length >= MAX_CARDS_PER_MEMORY) break;
4138
+ }
4139
+ return cards;
4140
+ }
4141
+ async function insertMemoryCardsForBatch(rows) {
4142
+ const cards = rows.flatMap(extractMemoryCards);
4143
+ if (cards.length === 0) return 0;
4144
+ const now = (/* @__PURE__ */ new Date()).toISOString();
4145
+ const client = getClient();
4146
+ const stmts = cards.map((card) => ({
4147
+ sql: `INSERT OR IGNORE INTO memory_cards
4148
+ (id, memory_id, agent_id, session_id, project_name, timestamp, card_type,
4149
+ subject, predicate, object, content, source_ref, confidence, active, created_at)
4150
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, 1, ?)`,
4151
+ args: [
4152
+ card.id,
4153
+ card.memory_id,
4154
+ card.agent_id,
4155
+ card.session_id,
4156
+ card.project_name,
4157
+ card.timestamp,
4158
+ card.card_type,
4159
+ card.subject,
4160
+ card.predicate,
4161
+ card.object,
4162
+ card.content,
4163
+ card.source_ref,
4164
+ card.confidence,
4165
+ now
4166
+ ]
4167
+ }));
4168
+ await client.batch(stmts, "write");
4169
+ return cards.length;
4170
+ }
4171
+ function buildMatchExpr(queryText) {
4172
+ const terms = queryText.toLowerCase().split(/\s+/).filter((t) => t.length >= 3).map((t) => t.replace(/[^a-z0-9_]/g, "")).filter((t) => t.length >= 3).slice(0, 12);
4173
+ if (terms.length === 0) return null;
4174
+ return terms.map((t) => `${t}*`).join(terms.length >= 3 ? " AND " : " OR ");
4175
+ }
4176
+ async function searchMemoryCards(queryText, agentId, options) {
4177
+ const limit = options?.limit ?? 10;
4178
+ const matchExpr = buildMatchExpr(queryText);
4179
+ if (!matchExpr) return [];
4180
+ let sql = `SELECT c.id, c.memory_id, c.agent_id, c.session_id, c.project_name,
4181
+ c.timestamp, c.card_type, c.content, c.source_ref, c.confidence
4182
+ FROM memory_cards c
4183
+ JOIN memory_cards_fts fts ON c.rowid = fts.rowid
4184
+ WHERE memory_cards_fts MATCH ?
4185
+ AND c.agent_id = ?
4186
+ AND COALESCE(c.active, 1) = 1`;
4187
+ const args = [matchExpr, agentId];
4188
+ if (options?.projectName) {
4189
+ sql += ` AND c.project_name = ?`;
4190
+ args.push(options.projectName);
4191
+ }
4192
+ if (options?.since) {
4193
+ sql += ` AND c.timestamp >= ?`;
4194
+ args.push(options.since);
4195
+ }
4196
+ sql += ` ORDER BY rank LIMIT ?`;
4197
+ args.push(limit);
4198
+ const result = await getClient().execute({ sql, args });
4199
+ return result.rows.map((row) => ({
4200
+ id: `card:${String(row.id)}`,
4201
+ agent_id: String(row.agent_id),
4202
+ agent_role: "memory_card",
4203
+ session_id: String(row.session_id),
4204
+ timestamp: String(row.timestamp),
4205
+ tool_name: `memory_card:${String(row.card_type)}`,
4206
+ project_name: row.project_name == null ? "" : String(row.project_name),
4207
+ has_error: false,
4208
+ raw_text: `[${String(row.card_type)}] ${String(row.content)}
4209
+ Source memory: ${String(row.source_ref ?? row.memory_id)}`,
4210
+ vector: [],
4211
+ importance: 6,
4212
+ status: "active",
4213
+ confidence: Number(row.confidence ?? 0.6),
4214
+ last_accessed: String(row.timestamp)
4215
+ }));
4216
+ }
4217
+ var MAX_CARDS_PER_MEMORY, MAX_SENTENCE_CHARS;
4218
+ var init_memory_cards = __esm({
4219
+ "src/lib/memory-cards.ts"() {
4220
+ "use strict";
4221
+ init_database();
4222
+ MAX_CARDS_PER_MEMORY = 6;
4223
+ MAX_SENTENCE_CHARS = 360;
4224
+ }
4225
+ });
4226
+
4024
4227
  // src/lib/store.ts
4025
4228
  var store_exports = {};
4026
4229
  __export(store_exports, {
@@ -4359,6 +4562,11 @@ async function flushBatch() {
4359
4562
  const globalClient = getClient();
4360
4563
  const globalStmts = batch.map(buildStmt);
4361
4564
  await globalClient.batch(globalStmts, "write");
4565
+ try {
4566
+ const { insertMemoryCardsForBatch: insertMemoryCardsForBatch2 } = await Promise.resolve().then(() => (init_memory_cards(), memory_cards_exports));
4567
+ await insertMemoryCardsForBatch2(batch);
4568
+ } catch {
4569
+ }
4362
4570
  schedulePostWriteMemoryHygiene(batch.map((row) => row.id));
4363
4571
  _pendingRecords.splice(0, batch.length);
4364
4572
  try {