@askexenow/exe-os 0.9.295 → 0.9.296

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 (279) hide show
  1. package/deploy/compose/cloudflared/config.yml.example +14 -9
  2. package/deploy/compose/docker-compose.yml +86 -8
  3. package/deploy/compose/sso-edge/default.conf.template +87 -0
  4. package/deploy/compose/sso-edge/entrypoint.sh +23 -0
  5. package/deploy/compose/sso-edge/sso-redirect.conf +63 -0
  6. package/deploy/stack-manifests/v0.9.json +1 -1
  7. package/dist/active-agent-AFX2FODG.js +28 -0
  8. package/dist/active-agent-E2IJA7YX.js +27 -0
  9. package/dist/agentic-ontology-A2YUZK5O.js +25 -0
  10. package/dist/assets/com.askexe.exed.plist +4 -1
  11. package/dist/backfill-metadata-OC7EOD5U.js +600 -0
  12. package/dist/behaviors-H5ZOVHDH.js +46 -0
  13. package/dist/bin/agentic-ontology-backfill.js +5 -5
  14. package/dist/bin/agentic-reflection-backfill.js +6 -6
  15. package/dist/bin/agentic-semantic-label.js +5 -5
  16. package/dist/bin/backfill-conversations.js +6 -6
  17. package/dist/bin/backfill-responses.js +6 -6
  18. package/dist/bin/backfill-vectors.js +8 -8
  19. package/dist/bin/bulk-sync-postgres.js +7 -7
  20. package/dist/bin/cc-doctor.js +4 -4
  21. package/dist/bin/cleanup-stale-review-tasks.js +11 -11
  22. package/dist/bin/cli.js +16 -16
  23. package/dist/bin/deferred-daemon-restart.js +1 -1
  24. package/dist/bin/exe-agent-config.js +2 -2
  25. package/dist/bin/exe-agent.js +4 -4
  26. package/dist/bin/exe-assign.js +8 -8
  27. package/dist/bin/exe-boot.js +21 -18
  28. package/dist/bin/exe-call.js +4 -4
  29. package/dist/bin/exe-cloud.js +7 -7
  30. package/dist/bin/exe-dispatch.js +11 -11
  31. package/dist/bin/exe-doctor.js +3 -2
  32. package/dist/bin/exe-export-behaviors.js +7 -7
  33. package/dist/bin/exe-forget.js +6 -6
  34. package/dist/bin/exe-gateway.js +7 -7
  35. package/dist/bin/exe-healthcheck.js +6 -4
  36. package/dist/bin/exe-heartbeat.js +11 -11
  37. package/dist/bin/exe-kill.js +14 -14
  38. package/dist/bin/exe-launch-agent.js +18 -18
  39. package/dist/bin/exe-new-employee.js +6 -6
  40. package/dist/bin/exe-pending-messages.js +12 -12
  41. package/dist/bin/exe-pending-notifications.js +11 -11
  42. package/dist/bin/exe-pending-reviews.js +11 -11
  43. package/dist/bin/exe-rename.js +4 -4
  44. package/dist/bin/exe-review.js +13 -13
  45. package/dist/bin/exe-search.js +5 -5
  46. package/dist/bin/exe-session-cleanup.js +16 -16
  47. package/dist/bin/exe-settings.js +39 -9
  48. package/dist/bin/exe-start-codex.js +11 -11
  49. package/dist/bin/exe-start-opencode.js +8 -8
  50. package/dist/bin/exe-status.js +12 -12
  51. package/dist/bin/exe-team.js +3 -3
  52. package/dist/bin/git-sweep.js +12 -12
  53. package/dist/bin/graph-backfill.js +4 -4
  54. package/dist/bin/graph-export.js +5 -5
  55. package/dist/bin/import-history.js +7 -7
  56. package/dist/bin/install-launchd.js +13 -6
  57. package/dist/bin/install.js +26 -14
  58. package/dist/bin/intercom-check.js +4 -4
  59. package/dist/bin/mcp-sessions.js +2 -2
  60. package/dist/bin/orchestration-metrics.js +4 -4
  61. package/dist/bin/postgres-agentic-reflection-backfill.js +2 -2
  62. package/dist/bin/postgres-agentic-semantic-backfill.js +1 -1
  63. package/dist/bin/scan-tasks.js +11 -11
  64. package/dist/bin/setup.js +1 -1
  65. package/dist/bin/shard-migrate.js +4 -4
  66. package/dist/bin/stack-update.js +2 -2
  67. package/dist/bin/vps-health-gate.js +1 -1
  68. package/dist/capability-cards-4USI7CUW.js +89 -0
  69. package/dist/capacity-monitor-WLCBTEYR.js +51 -0
  70. package/dist/catchup-brief-ZR3NX6LZ.js +175 -0
  71. package/dist/chunk-22TVSRQQ.js +226 -0
  72. package/dist/chunk-2E43UXRH.js +395 -0
  73. package/dist/chunk-2PIGT6UJ.js +460 -0
  74. package/dist/chunk-3XTMW2MZ.js +535 -0
  75. package/dist/chunk-465PQFTH.js +262 -0
  76. package/dist/chunk-5CCXU2AW.js +129 -0
  77. package/dist/chunk-5D6MPWR7.js +1094 -0
  78. package/dist/chunk-5Q4MR6SL.js +123 -0
  79. package/dist/chunk-6327RBWR.js +345 -0
  80. package/dist/chunk-6MZZREZY.js +199 -0
  81. package/dist/chunk-7DI2Q4O5.js +1186 -0
  82. package/dist/chunk-7PW5VNIY.js +122 -0
  83. package/dist/chunk-7T7Y56HW.js +43 -0
  84. package/dist/chunk-7UHCWCLT.js +128 -0
  85. package/dist/chunk-A2ZUMF6L.js +1350 -0
  86. package/dist/chunk-AKV44JEH.js +185 -0
  87. package/dist/chunk-ANHWGX5N.js +735 -0
  88. package/dist/chunk-BQ3P4TKD.js +97 -0
  89. package/dist/chunk-BUZMT3KZ.js +604 -0
  90. package/dist/chunk-C2SBESBO.js +210 -0
  91. package/dist/chunk-CLSXZUZW.js +51 -0
  92. package/dist/chunk-CONHLVAR.js +1079 -0
  93. package/dist/chunk-D3WTZPFX.js +456 -0
  94. package/dist/chunk-DE6SOIYL.js +197 -0
  95. package/dist/chunk-EIVNMA3Q.js +284 -0
  96. package/dist/chunk-EJIF4FNT.js +12 -0
  97. package/dist/chunk-FDFOW564.js +171 -0
  98. package/dist/chunk-GZUBJ5EC.js +127 -0
  99. package/dist/chunk-HGZITN22.js +105 -0
  100. package/dist/chunk-HSRKDU6X.js +362 -0
  101. package/dist/chunk-IIEN2PHV.js +85 -0
  102. package/dist/chunk-JQ56VLMM.js +567 -0
  103. package/dist/chunk-JVHHXRFY.js +280 -0
  104. package/dist/chunk-JXCXGZ3S.js +55 -0
  105. package/dist/chunk-K5ZO532Q.js +4388 -0
  106. package/dist/chunk-K6CAAMXF.js +97 -0
  107. package/dist/chunk-KA26YTNU.js +81 -0
  108. package/dist/chunk-KMUW5C3R.js +381 -0
  109. package/dist/chunk-KOO3J5PV.js +20 -0
  110. package/dist/chunk-LSV7OFIH.js +290 -0
  111. package/dist/chunk-LSVFDVNY.js +1158 -0
  112. package/dist/chunk-LXDQTW32.js +230 -0
  113. package/dist/chunk-MEP7OUVZ.js +181 -0
  114. package/dist/chunk-MN2B2LKS.js +240 -0
  115. package/dist/chunk-N2EAYPYQ.js +1352 -0
  116. package/dist/chunk-N7I2A667.js +70 -0
  117. package/dist/chunk-NLZHVIOP.js +630 -0
  118. package/dist/chunk-NUH5TRZL.js +227 -0
  119. package/dist/chunk-OAHEIH3G.js +167 -0
  120. package/dist/chunk-OBHRQGCK.js +58 -0
  121. package/dist/chunk-ODFA7B2V.js +54 -0
  122. package/dist/chunk-OSNUP45F.js +731 -0
  123. package/dist/chunk-OTPRHBTO.js +33 -0
  124. package/dist/chunk-P6MUA4QU.js +157 -0
  125. package/dist/chunk-PGIOFKSK.js +2093 -0
  126. package/dist/chunk-PSE7VHWK.js +50 -0
  127. package/dist/chunk-QIFUVZFW.js +331 -0
  128. package/dist/chunk-RDPXKTVK.js +221 -0
  129. package/dist/chunk-RKYTYJGB.js +76 -0
  130. package/dist/chunk-RXLR6EFM.js +348 -0
  131. package/dist/chunk-SDB67PQJ.js +159 -0
  132. package/dist/chunk-SF2T7MP3.js +402 -0
  133. package/dist/chunk-SLU3FRFQ.js +2133 -0
  134. package/dist/chunk-SNDZJ5IV.js +214 -0
  135. package/dist/chunk-STEEAABW.js +448 -0
  136. package/dist/chunk-TUTWNHIQ.js +244 -0
  137. package/dist/chunk-UDP35QBR.js +30 -0
  138. package/dist/chunk-UKFHNJBI.js +85 -0
  139. package/dist/chunk-VC2DTK2X.js +382 -0
  140. package/dist/chunk-VRRAE5JX.js +836 -0
  141. package/dist/chunk-VVJTBQPR.js +38 -0
  142. package/dist/chunk-W3EQ362K.js +581 -0
  143. package/dist/chunk-WHIXIFHC.js +2242 -0
  144. package/dist/chunk-WRNGJJNR.js +377 -0
  145. package/dist/chunk-WUKHLCBE.js +3313 -0
  146. package/dist/chunk-WVPLHGDG.js +150 -0
  147. package/dist/chunk-XJZBSTL5.js +204 -0
  148. package/dist/chunk-Y3PMNUM5.js +304 -0
  149. package/dist/chunk-YHVS4QOV.js +14597 -0
  150. package/dist/chunk-YJ2OYAOC.js +668 -0
  151. package/dist/chunk-YYAD2GXX.js +128 -0
  152. package/dist/chunk-ZQML7EWE.js +333 -0
  153. package/dist/co-activation-XJLH46OX.js +74 -0
  154. package/dist/co-occurrence-GNN2X526.js +95 -0
  155. package/dist/code-context-index-OCPRLFG5.js +30 -0
  156. package/dist/core-memory-J4W2IYOF.js +110 -0
  157. package/dist/crdt-sync-QCBTSHIH.js +33 -0
  158. package/dist/crm-webhook-EM442VUW.js +10 -0
  159. package/dist/cto-delegation-gate-MLJMVHBK.js +280 -0
  160. package/dist/daemon-orchestration-2VNLZVTW.js +139 -0
  161. package/dist/db-backup-VUGFTPJ4.js +43 -0
  162. package/dist/doc-graph-extractor-PNRSFPSS.js +133 -0
  163. package/dist/dreaming-SK5VEQRF.js +34 -0
  164. package/dist/entity-boost-TQWWJUC2.js +375 -0
  165. package/dist/exe-drift-N34UPO7S.js +70 -0
  166. package/dist/exe-export-KACBKGVV.js +77 -0
  167. package/dist/exe-import-GXGDWACG.js +80 -0
  168. package/dist/exe-key-XPDOZBWW.js +673 -0
  169. package/dist/exe-snapshot-32GQKGQ5.js +338 -0
  170. package/dist/fast-db-init-F3TDD5VV.js +7 -0
  171. package/dist/gateway/index.js +8 -8
  172. package/dist/git-staleness-J45WNYRF.js +112 -0
  173. package/dist/git-task-sweep-BTGVQPFB.js +42 -0
  174. package/dist/global-procedures-6JCQWU4D.js +22 -0
  175. package/dist/graph-auto-extract-3ZQNXTPC.js +183 -0
  176. package/dist/hooks/bug-report-worker.js +13 -13
  177. package/dist/hooks/codex-stop-task-finalizer.js +13 -13
  178. package/dist/hooks/commit-complete.js +13 -13
  179. package/dist/hooks/error-recall.js +6 -6
  180. package/dist/hooks/exe-heartbeat-hook.js +3 -3
  181. package/dist/hooks/ingest-worker.js +3 -3
  182. package/dist/hooks/ingest.js +6 -6
  183. package/dist/hooks/instructions-loaded.js +4 -4
  184. package/dist/hooks/manifest.json +20 -20
  185. package/dist/hooks/notification.js +4 -4
  186. package/dist/hooks/post-compact.js +12 -12
  187. package/dist/hooks/post-tool-combined.js +6 -6
  188. package/dist/hooks/pre-compact.js +16 -16
  189. package/dist/hooks/pre-tool-use.js +16 -16
  190. package/dist/hooks/prompt-submit.js +24 -24
  191. package/dist/hooks/session-end.js +21 -21
  192. package/dist/hooks/session-start.js +12 -12
  193. package/dist/hooks/stop.js +19 -19
  194. package/dist/hooks/subagent-stop.js +12 -12
  195. package/dist/hooks/summary-worker.js +19 -19
  196. package/dist/index.js +19 -19
  197. package/dist/installer-5VPFY7SB.js +298 -0
  198. package/dist/installer-OENFPMA2.js +344 -0
  199. package/dist/installer-OIX4QOG5.js +40 -0
  200. package/dist/lib/cloud-sync.js +7 -7
  201. package/dist/lib/consolidation.js +6 -5
  202. package/dist/lib/database.js +2 -2
  203. package/dist/lib/db-daemon-client.js +2 -2
  204. package/dist/lib/db.js +2 -2
  205. package/dist/lib/embed-worker.js +1 -0
  206. package/dist/lib/embedder.js +7 -3
  207. package/dist/lib/employee-templates.js +4 -4
  208. package/dist/lib/employees.js +2 -2
  209. package/dist/lib/exe-daemon-client.js +2 -2
  210. package/dist/lib/exe-daemon.js +160 -79
  211. package/dist/lib/hybrid-search.js +5 -5
  212. package/dist/lib/identity.js +2 -2
  213. package/dist/lib/messaging.js +11 -11
  214. package/dist/lib/reminders.js +3 -3
  215. package/dist/lib/schedules.js +5 -5
  216. package/dist/lib/session-registry.js +4 -4
  217. package/dist/lib/skill-learning.js +6 -6
  218. package/dist/lib/store.js +4 -4
  219. package/dist/lib/task-router.js +3 -3
  220. package/dist/lib/tasks.js +12 -12
  221. package/dist/lib/tmux-routing.js +12 -10
  222. package/dist/lib/tmux-transport.js +1 -1
  223. package/dist/lib/token-spend.js +3 -3
  224. package/dist/lib/transport.js +2 -2
  225. package/dist/mcp/register-tools.js +62 -61
  226. package/dist/mcp/server.js +63 -62
  227. package/dist/mcp/tools/complete-reminder.js +4 -4
  228. package/dist/mcp/tools/create-reminder.js +4 -4
  229. package/dist/mcp/tools/create-task.js +14 -14
  230. package/dist/mcp/tools/deactivate-behavior.js +7 -7
  231. package/dist/mcp/tools/list-reminders.js +4 -4
  232. package/dist/mcp/tools/list-tasks.js +14 -14
  233. package/dist/mcp/tools/send-message.js +13 -13
  234. package/dist/mcp/tools/update-task.js +13 -13
  235. package/dist/mcp-http-config-PQTOLCTP.js +29 -0
  236. package/dist/memory-cards-4RVDZIY2.js +180 -0
  237. package/dist/memory-graph-extractor-L6YC7G4M.js +22 -0
  238. package/dist/memory-poisoning-defense-4YVJYH4G.js +224 -0
  239. package/dist/memory-queue-client-MVAUOZNJ.js +16 -0
  240. package/dist/memory-reflection-SHHDQNOH.js +244 -0
  241. package/dist/message-queue-client-DCKZT6X2.js +92 -0
  242. package/dist/notifications-JFR3G42W.js +47 -0
  243. package/dist/orchestration-events-MGCGPTDN.js +27 -0
  244. package/dist/orchestrator-DAFL2YZB.js +35 -0
  245. package/dist/pipeline-router-WWSZVPCH.js +15 -0
  246. package/dist/plan-limits-C7XCSDZC.js +28 -0
  247. package/dist/project-boot-N3NTBVLE.js +299 -0
  248. package/dist/projection-worker-MTPAPCWX.js +1084 -0
  249. package/dist/prospective-memory-BTIVUJSB.js +232 -0
  250. package/dist/reranker-UA6WVESJ.js +19 -0
  251. package/dist/retrieval-health-7XNZJEBF.js +12 -0
  252. package/dist/review-polling-4ALGMXC3.js +126 -0
  253. package/dist/runtime/index.js +13 -13
  254. package/dist/self-query-router-MROFQLQB.js +192 -0
  255. package/dist/session-events-CK44XOU4.js +38 -0
  256. package/dist/session-kill-telemetry-MT6ITDOG.js +31 -0
  257. package/dist/session-scope-3XDBWV65.js +88 -0
  258. package/dist/setup-wizard-X6DOD7MC.js +12 -0
  259. package/dist/skill-refinement-G2CCY3GM.js +159 -0
  260. package/dist/stack-update-JF7F56AS.js +84 -0
  261. package/dist/steward-gate-YF2CYXE7.js +15 -0
  262. package/dist/task-enforcement-YN6HK7NE.js +506 -0
  263. package/dist/task-scope-CVK6ISCZ.js +37 -0
  264. package/dist/tasks-crud-NTNET4JE.js +79 -0
  265. package/dist/tasks-notify-4LJVFPCV.js +40 -0
  266. package/dist/tasks-review-3V4WOIRG.js +49 -0
  267. package/dist/telemetry-upload-5PNUKGTM.js +741 -0
  268. package/dist/token-budget-E46G7ZAQ.js +86 -0
  269. package/dist/tool-capability-index-JDSMKJER.js +10 -0
  270. package/dist/tool-telemetry-J3NLS3LJ.js +17 -0
  271. package/dist/tui/App.js +18 -18
  272. package/dist/tui-data-6DOMUUCM.js +260 -0
  273. package/dist/wiki-acl-5UK37LKF.js +111 -0
  274. package/dist/worker-gate-FM7AEC7G.js +21 -0
  275. package/dist/workflow-engine-2EDUHUIY.js +28 -0
  276. package/dist/worktree-7YKKJIYR.js +28 -0
  277. package/dist/worktree-sweep-C3ELFGDN.js +21 -0
  278. package/package.json +1 -1
  279. package/release-notes.json +23 -23
@@ -0,0 +1,377 @@
1
+ import {
2
+ extractMemoryGraph
3
+ } from "./chunk-SF2T7MP3.js";
4
+ import {
5
+ flushBatch,
6
+ writeMemory
7
+ } from "./chunk-VRRAE5JX.js";
8
+ import {
9
+ extractKeywords
10
+ } from "./chunk-CHCA3ZM2.js";
11
+ import {
12
+ getActiveAgent
13
+ } from "./chunk-AKV44JEH.js";
14
+
15
+ // src/mcp/tools/import-conversations.ts
16
+ import { z } from "zod";
17
+ import crypto from "crypto";
18
+ import { readdir, readFile, stat } from "fs/promises";
19
+ import path, { resolve, normalize } from "path";
20
+ import os from "os";
21
+
22
+ // src/lib/conversation-parser.ts
23
+ function detectFormat(content, filePath) {
24
+ const lower = filePath.toLowerCase();
25
+ if (lower.endsWith(".md") || lower.endsWith(".markdown")) return "markdown";
26
+ const trimmed = content.trimStart();
27
+ if (!trimmed.startsWith("[") && !trimmed.startsWith("{")) return "markdown";
28
+ try {
29
+ const parsed = JSON.parse(trimmed);
30
+ if (Array.isArray(parsed) && parsed.length > 0) {
31
+ const first = parsed[0];
32
+ if (first && typeof first === "object") {
33
+ if ("mapping" in first && typeof first.mapping === "object") return "chatgpt";
34
+ if ("chat_messages" in first) return "claude";
35
+ if ("uuid" in first && "name" in first) return "claude";
36
+ }
37
+ }
38
+ return "generic";
39
+ } catch {
40
+ return "markdown";
41
+ }
42
+ }
43
+ function extractChatGPTContent(node) {
44
+ const parts = node.message?.content?.parts;
45
+ if (!Array.isArray(parts)) return "";
46
+ return parts.filter((p) => typeof p === "string").join("\n").trim();
47
+ }
48
+ function parseChatGPT(content, sourceFile = "conversations.json") {
49
+ const data = JSON.parse(content);
50
+ if (!Array.isArray(data)) return [];
51
+ return data.map((conv) => {
52
+ const title = conv.title ?? "Untitled";
53
+ const messages = [];
54
+ if (conv.mapping) {
55
+ const nodes = Object.values(conv.mapping).filter((n) => n.message?.content?.parts && extractChatGPTContent(n).length > 0).sort((a, b) => (a.message?.create_time ?? 0) - (b.message?.create_time ?? 0));
56
+ for (const node of nodes) {
57
+ const role = node.message?.author?.role ?? "unknown";
58
+ const text = extractChatGPTContent(node);
59
+ if (!text) continue;
60
+ const createTime = node.message?.create_time;
61
+ messages.push({
62
+ role,
63
+ content: text,
64
+ timestamp: createTime ? new Date(createTime * 1e3).toISOString() : void 0
65
+ });
66
+ }
67
+ }
68
+ return { title, messages, sourceFile };
69
+ }).filter((c) => c.messages.length > 0);
70
+ }
71
+ function parseClaude(content, sourceFile = "conversations.json") {
72
+ const data = JSON.parse(content);
73
+ if (!Array.isArray(data)) return [];
74
+ return data.map((conv) => {
75
+ const title = conv.name ?? conv.uuid ?? "Untitled";
76
+ const messages = [];
77
+ if (conv.chat_messages) {
78
+ for (const msg of conv.chat_messages) {
79
+ let text = msg.text ?? "";
80
+ if (!text && Array.isArray(msg.content)) {
81
+ text = msg.content.filter((c) => c.type === "text" && c.text).map((c) => c.text).join("\n");
82
+ }
83
+ if (!text.trim()) continue;
84
+ messages.push({
85
+ role: msg.sender ?? "unknown",
86
+ content: text.trim(),
87
+ timestamp: msg.created_at
88
+ });
89
+ }
90
+ }
91
+ return { title, messages, sourceFile };
92
+ }).filter((c) => c.messages.length > 0);
93
+ }
94
+ function parseGeneric(content, sourceFile = "conversations.json") {
95
+ const trimmed = content.trimStart();
96
+ try {
97
+ const data = JSON.parse(trimmed);
98
+ if (Array.isArray(data)) {
99
+ if (data.length > 0 && typeof data[0] === "object" && ("role" in data[0] || "content" in data[0])) {
100
+ const messages = data.map((msg) => ({
101
+ role: msg.role ?? "unknown",
102
+ content: (msg.content ?? msg.text ?? "").trim(),
103
+ timestamp: msg.timestamp ?? msg.created_at
104
+ })).filter((m) => m.content.length > 0);
105
+ if (messages.length > 0) {
106
+ return [{ title: sourceFile, messages, sourceFile }];
107
+ }
108
+ }
109
+ }
110
+ } catch {
111
+ }
112
+ return parseMarkdown(content, sourceFile);
113
+ }
114
+ function parseMarkdown(content, sourceFile = "conversation.md") {
115
+ const lines = content.split("\n");
116
+ const conversations = [];
117
+ let currentTitle = sourceFile;
118
+ let currentMessages = [];
119
+ let currentRole = "user";
120
+ let currentContent = [];
121
+ function flushMessage() {
122
+ const text = currentContent.join("\n").trim();
123
+ if (text) {
124
+ currentMessages.push({ role: currentRole, content: text });
125
+ }
126
+ currentContent = [];
127
+ }
128
+ function flushConversation() {
129
+ flushMessage();
130
+ if (currentMessages.length > 0) {
131
+ conversations.push({ title: currentTitle, messages: [...currentMessages], sourceFile });
132
+ }
133
+ currentMessages = [];
134
+ }
135
+ for (const line of lines) {
136
+ const h1Match = line.match(/^# (.+)$/);
137
+ if (h1Match) {
138
+ flushConversation();
139
+ currentTitle = h1Match[1];
140
+ continue;
141
+ }
142
+ const roleMatch = line.match(/^(?:##\s+|\*\*)(user|assistant|human|system|ai|claude|gpt)(?:\*\*)?:?\s*/i);
143
+ if (roleMatch) {
144
+ flushMessage();
145
+ const raw = roleMatch[1].toLowerCase();
146
+ currentRole = raw === "human" || raw === "user" ? "user" : raw === "assistant" || raw === "ai" || raw === "claude" || raw === "gpt" ? "assistant" : raw;
147
+ continue;
148
+ }
149
+ currentContent.push(line);
150
+ }
151
+ flushConversation();
152
+ return conversations;
153
+ }
154
+ function conversationsToMemories(convos, format = "generic") {
155
+ const memories = [];
156
+ for (const convo of convos) {
157
+ for (const msg of convo.messages) {
158
+ if (msg.content.length < 10) continue;
159
+ memories.push({
160
+ text: msg.content,
161
+ kind: "conversation",
162
+ metadata: {
163
+ source_type: "import",
164
+ source_format: format,
165
+ source_path: convo.sourceFile,
166
+ conversation_title: convo.title,
167
+ role: msg.role,
168
+ original_timestamp: msg.timestamp
169
+ }
170
+ });
171
+ }
172
+ }
173
+ return memories;
174
+ }
175
+
176
+ // src/mcp/tools/import-conversations.ts
177
+ var DEFAULT_CAP = 5e4;
178
+ var YIELD_INTERVAL = 50;
179
+ async function readInputFiles(inputPath) {
180
+ const files = [];
181
+ const info = await stat(inputPath);
182
+ if (info.isDirectory()) {
183
+ const entries = await readdir(inputPath);
184
+ for (const entry of entries) {
185
+ const fullPath = path.join(inputPath, entry);
186
+ const entryStat = await stat(fullPath);
187
+ if (entryStat.isFile() && (entry.endsWith(".json") || entry.endsWith(".md") || entry.endsWith(".txt"))) {
188
+ files.push({ path: fullPath, content: await readFile(fullPath, "utf-8") });
189
+ }
190
+ }
191
+ } else {
192
+ files.push({ path: inputPath, content: await readFile(inputPath, "utf-8") });
193
+ }
194
+ return files;
195
+ }
196
+ function parseFile(content, filePath, requestedFormat) {
197
+ const format = requestedFormat && requestedFormat !== "auto" ? requestedFormat : detectFormat(content, filePath);
198
+ let conversations;
199
+ switch (format) {
200
+ case "chatgpt":
201
+ conversations = parseChatGPT(content, filePath);
202
+ break;
203
+ case "claude":
204
+ conversations = parseClaude(content, filePath);
205
+ break;
206
+ case "markdown":
207
+ conversations = parseMarkdown(content, filePath);
208
+ break;
209
+ case "generic":
210
+ default:
211
+ conversations = parseGeneric(content, filePath);
212
+ break;
213
+ }
214
+ return { format, conversations };
215
+ }
216
+ function registerImportConversations(server) {
217
+ server.registerTool(
218
+ "import_conversations",
219
+ {
220
+ title: "Import Conversations",
221
+ description: "Import ChatGPT, Claude, or generic conversation exports into memory. Parses, extracts keywords + graph entities, stores with proper metadata. Deduplicates via content_hash. Supports JSON, markdown, and directory input.",
222
+ inputSchema: {
223
+ path: z.string().describe("File or directory path to import"),
224
+ format: z.enum(["chatgpt", "claude", "generic", "markdown", "auto"]).default("auto").describe("Source format (default: auto-detect)"),
225
+ project_name: z.string().optional().describe("Project name for imported memories"),
226
+ dry_run: z.boolean().default(false).describe("Parse and report counts without storing"),
227
+ force: z.boolean().default(false).describe("Override 50K memory cap"),
228
+ agent_id: z.string().optional().describe("Agent to attribute memories to (default: current agent)")
229
+ }
230
+ },
231
+ async ({ path: inputPath, format: requestedFormat, project_name, dry_run, force, agent_id: overrideAgentId }) => {
232
+ const { agentId: currentAgentId, agentRole } = getActiveAgent();
233
+ const agentId = overrideAgentId ?? currentAgentId;
234
+ const projectName = project_name ?? "imported";
235
+ const resolvedPath = resolve(normalize(inputPath));
236
+ const home = process.env.HOME || os.homedir();
237
+ const blocked = ["/etc", "/root", "/var", "/usr", "/sys", "/proc"];
238
+ const sensitiveHome = [".ssh", ".gnupg", ".aws", ".config/gcloud"];
239
+ for (const dir of blocked) {
240
+ if (resolvedPath.startsWith(dir + "/") || resolvedPath === dir) {
241
+ return {
242
+ content: [{ type: "text", text: `Access denied: cannot read files from ${dir}` }],
243
+ isError: true
244
+ };
245
+ }
246
+ }
247
+ for (const rel of sensitiveHome) {
248
+ const full = resolve(home, rel);
249
+ if (resolvedPath.startsWith(full + "/") || resolvedPath === full) {
250
+ return {
251
+ content: [{ type: "text", text: `Access denied: cannot read files from ~/${rel}` }],
252
+ isError: true
253
+ };
254
+ }
255
+ }
256
+ let files;
257
+ try {
258
+ files = await readInputFiles(inputPath);
259
+ } catch (err) {
260
+ return {
261
+ content: [{ type: "text", text: `Error reading ${inputPath}: ${err instanceof Error ? err.message : String(err)}` }],
262
+ isError: true
263
+ };
264
+ }
265
+ if (files.length === 0) {
266
+ return {
267
+ content: [{ type: "text", text: `No importable files found at ${inputPath}` }],
268
+ isError: true
269
+ };
270
+ }
271
+ const allMemories = [];
272
+ let conversationCount = 0;
273
+ const formatCounts = {};
274
+ for (const file of files) {
275
+ const { format, conversations } = parseFile(file.content, file.path, requestedFormat);
276
+ formatCounts[format] = (formatCounts[format] ?? 0) + conversations.length;
277
+ conversationCount += conversations.length;
278
+ const memories = conversationsToMemories(conversations, format);
279
+ allMemories.push(...memories);
280
+ }
281
+ if (allMemories.length > DEFAULT_CAP && !force) {
282
+ return {
283
+ content: [{
284
+ type: "text",
285
+ text: `Import would create ${allMemories.length} memories (cap: ${DEFAULT_CAP}). Use force=true to override. Files: ${files.length}, Conversations: ${conversationCount}`
286
+ }],
287
+ isError: true
288
+ };
289
+ }
290
+ if (dry_run) {
291
+ const formatSummary2 = Object.entries(formatCounts).map(([f, c]) => `${f}: ${c}`).join(", ");
292
+ return {
293
+ content: [{
294
+ type: "text",
295
+ text: `[DRY RUN] Would import:
296
+ Files: ${files.length}
297
+ Conversations: ${conversationCount} (${formatSummary2})
298
+ Memories: ${allMemories.length}
299
+ Agent: ${agentId}
300
+ Project: ${projectName}`
301
+ }]
302
+ };
303
+ }
304
+ let stored = 0;
305
+ let skipped = 0;
306
+ for (let i = 0; i < allMemories.length; i++) {
307
+ const memory = allMemories[i];
308
+ try {
309
+ const keywords = extractKeywords(memory.text);
310
+ const keywordsStr = keywords.length > 0 ? keywords.join(" ") : null;
311
+ let entities = null;
312
+ try {
313
+ const graph = extractMemoryGraph(memory.text, agentId, projectName, "import_conversations");
314
+ if (graph.entities.length > 0) {
315
+ entities = JSON.stringify(graph.entities.map((e) => e.name));
316
+ }
317
+ } catch {
318
+ }
319
+ const id = crypto.randomUUID();
320
+ await writeMemory({
321
+ id,
322
+ agent_id: agentId,
323
+ agent_role: agentRole,
324
+ session_id: process.env.SESSION_ID ?? "import",
325
+ timestamp: memory.metadata.original_timestamp ?? (/* @__PURE__ */ new Date()).toISOString(),
326
+ tool_name: "import_conversations",
327
+ project_name: projectName,
328
+ has_error: false,
329
+ raw_text: memory.text,
330
+ vector: null,
331
+ // Backfill later
332
+ source_path: memory.metadata.source_path,
333
+ source_type: "import",
334
+ memory_type: "conversation",
335
+ tier: 3,
336
+ // Passive ingestion
337
+ importance: memory.metadata.role === "assistant" ? 5 : 4,
338
+ intent: "observation",
339
+ domain: projectName,
340
+ referenced_entities: entities,
341
+ keywords: keywordsStr,
342
+ valid_from: memory.metadata.original_timestamp ?? void 0
343
+ });
344
+ stored++;
345
+ } catch {
346
+ skipped++;
347
+ }
348
+ if ((i + 1) % YIELD_INTERVAL === 0) {
349
+ await flushBatch();
350
+ await new Promise((r) => setTimeout(r, 0));
351
+ if ((i + 1) % (YIELD_INTERVAL * 10) === 0) {
352
+ process.stderr.write(`[import] ${i + 1}/${allMemories.length} memories processed, ${stored} stored
353
+ `);
354
+ }
355
+ }
356
+ }
357
+ await flushBatch();
358
+ const formatSummary = Object.entries(formatCounts).map(([f, c]) => `${f}: ${c}`).join(", ");
359
+ return {
360
+ content: [{
361
+ type: "text",
362
+ text: `Import complete.
363
+ Files: ${files.length}
364
+ Conversations: ${conversationCount} (${formatSummary})
365
+ Memories stored: ${stored}
366
+ Duplicates skipped: ${skipped}
367
+ Agent: ${agentId}
368
+ Project: ${projectName}`
369
+ }]
370
+ };
371
+ }
372
+ );
373
+ }
374
+
375
+ export {
376
+ registerImportConversations
377
+ };