@aria-cli/tools 1.0.12 → 1.0.14

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 (233) hide show
  1. package/dist/index.js +378 -70
  2. package/dist/network-runtime/index.js +8 -12
  3. package/dist-cjs/index.js +400 -435
  4. package/dist-cjs/network-runtime/index.js +8 -172
  5. package/package.json +8 -6
  6. package/dist/.tsbuildinfo +0 -1
  7. package/dist/ask-user-interaction.js +0 -22
  8. package/dist/cache/web-cache.js +0 -66
  9. package/dist/definitions/arion.js +0 -104
  10. package/dist/definitions/browser/browser.js +0 -418
  11. package/dist/definitions/browser/index.js +0 -4
  12. package/dist/definitions/browser/pw-downloads.js +0 -114
  13. package/dist/definitions/browser/pw-interactions.js +0 -199
  14. package/dist/definitions/browser/pw-responses.js +0 -76
  15. package/dist/definitions/browser/pw-session.js +0 -310
  16. package/dist/definitions/browser/pw-shared.js +0 -66
  17. package/dist/definitions/browser/pw-snapshot.js +0 -301
  18. package/dist/definitions/browser/pw-state.js +0 -62
  19. package/dist/definitions/browser/types.js +0 -4
  20. package/dist/definitions/code-intelligence.js +0 -470
  21. package/dist/definitions/core.js +0 -109
  22. package/dist/definitions/delegation.js +0 -512
  23. package/dist/definitions/deploy.js +0 -65
  24. package/dist/definitions/filesystem.js +0 -196
  25. package/dist/definitions/frg.js +0 -63
  26. package/dist/definitions/index.js +0 -20
  27. package/dist/definitions/memory.js +0 -123
  28. package/dist/definitions/messaging.js +0 -625
  29. package/dist/definitions/meta.js +0 -349
  30. package/dist/definitions/network.js +0 -159
  31. package/dist/definitions/outlook.js +0 -277
  32. package/dist/definitions/patch/apply-patch.js +0 -184
  33. package/dist/definitions/patch/fuzzy-match.js +0 -166
  34. package/dist/definitions/patch/index.js +0 -1
  35. package/dist/definitions/patch/patch-parser.js +0 -207
  36. package/dist/definitions/patch/sandbox-paths.js +0 -105
  37. package/dist/definitions/process/index.js +0 -4
  38. package/dist/definitions/process/process-registry.js +0 -213
  39. package/dist/definitions/process/process.js +0 -386
  40. package/dist/definitions/process/pty-keys.js +0 -254
  41. package/dist/definitions/process/session-slug.js +0 -142
  42. package/dist/definitions/quip.js +0 -195
  43. package/dist/definitions/search.js +0 -60
  44. package/dist/definitions/session-history.js +0 -69
  45. package/dist/definitions/shell.js +0 -181
  46. package/dist/definitions/slack.js +0 -180
  47. package/dist/definitions/web.js +0 -109
  48. package/dist/executors/apply-patch.js +0 -901
  49. package/dist/executors/arion.js +0 -119
  50. package/dist/executors/code-intelligence.js +0 -882
  51. package/dist/executors/deploy.js +0 -848
  52. package/dist/executors/filesystem.js +0 -1122
  53. package/dist/executors/frg-freshness.js +0 -576
  54. package/dist/executors/frg.js +0 -298
  55. package/dist/executors/index.js +0 -46
  56. package/dist/executors/learning-meta.js +0 -1146
  57. package/dist/executors/lsp-client.js +0 -296
  58. package/dist/executors/memory.js +0 -750
  59. package/dist/executors/meta.js +0 -220
  60. package/dist/executors/process-registry.js +0 -465
  61. package/dist/executors/pty-session-store.js +0 -30
  62. package/dist/executors/pty.js +0 -271
  63. package/dist/executors/restart.js +0 -119
  64. package/dist/executors/search-freshness.js +0 -195
  65. package/dist/executors/search-types.js +0 -52
  66. package/dist/executors/search.js +0 -66
  67. package/dist/executors/self-diagnose.js +0 -398
  68. package/dist/executors/session-history.js +0 -283
  69. package/dist/executors/shell-safety.js +0 -473
  70. package/dist/executors/shell.js +0 -954
  71. package/dist/executors/utils.js +0 -33
  72. package/dist/executors/web.js +0 -542
  73. package/dist/extraction/content-extraction.js +0 -235
  74. package/dist/extraction/index.js +0 -4
  75. package/dist/headless-control-contract.js +0 -967
  76. package/dist/local-control-http-auth.js +0 -2
  77. package/dist/mcp/client.js +0 -181
  78. package/dist/mcp/connection.js +0 -480
  79. package/dist/mcp/index.js +0 -10
  80. package/dist/mcp/jsonrpc.js +0 -144
  81. package/dist/mcp/types.js +0 -7
  82. package/dist/network-control-adapter.js +0 -72
  83. package/dist/network-runtime/address-types.js +0 -165
  84. package/dist/network-runtime/db-owner-fencing.js +0 -69
  85. package/dist/network-runtime/delivery-receipts.js +0 -267
  86. package/dist/network-runtime/direct-endpoint-authority.js +0 -25
  87. package/dist/network-runtime/local-control-contract.js +0 -627
  88. package/dist/network-runtime/node-store-contract.js +0 -34
  89. package/dist/network-runtime/pair-route-contract.js +0 -77
  90. package/dist/network-runtime/peer-capabilities.js +0 -28
  91. package/dist/network-runtime/peer-principal-ref.js +0 -12
  92. package/dist/network-runtime/peer-state-machine.js +0 -121
  93. package/dist/network-runtime/protocol-schemas.js +0 -205
  94. package/dist/network-runtime/runtime-bootstrap-contract.js +0 -60
  95. package/dist/outlook/desktop-session.js +0 -279
  96. package/dist/policy.js +0 -149
  97. package/dist/providers/brave.js +0 -62
  98. package/dist/providers/duckduckgo.js +0 -176
  99. package/dist/providers/exa.js +0 -63
  100. package/dist/providers/firecrawl.js +0 -55
  101. package/dist/providers/index.js +0 -7
  102. package/dist/providers/jina.js +0 -49
  103. package/dist/providers/router.js +0 -96
  104. package/dist/providers/search-provider.js +0 -32
  105. package/dist/providers/tavily.js +0 -54
  106. package/dist/quip/desktop-session.js +0 -317
  107. package/dist/registry/index.js +0 -1
  108. package/dist/registry/registry.js +0 -756
  109. package/dist/runtime-socket-local-control-client.js +0 -330
  110. package/dist/security/dns-normalization.js +0 -19
  111. package/dist/security/dns-pinning.js +0 -123
  112. package/dist/security/external-content.js +0 -91
  113. package/dist/security/ssrf.js +0 -181
  114. package/dist/slack/desktop-session.js +0 -324
  115. package/dist/tool-factory.js +0 -47
  116. package/dist/types.js +0 -7
  117. package/dist/utils/retry.js +0 -132
  118. package/dist/utils/safe-parse-json.js +0 -160
  119. package/dist/utils/url.js +0 -19
  120. package/dist-cjs/.tsbuildinfo +0 -1
  121. package/dist-cjs/ask-user-interaction.js +0 -27
  122. package/dist-cjs/cache/web-cache.js +0 -70
  123. package/dist-cjs/definitions/arion.js +0 -107
  124. package/dist-cjs/definitions/browser/browser.js +0 -421
  125. package/dist-cjs/definitions/browser/index.js +0 -8
  126. package/dist-cjs/definitions/browser/pw-downloads.js +0 -117
  127. package/dist-cjs/definitions/browser/pw-interactions.js +0 -213
  128. package/dist-cjs/definitions/browser/pw-responses.js +0 -84
  129. package/dist-cjs/definitions/browser/pw-session.js +0 -326
  130. package/dist-cjs/definitions/browser/pw-shared.js +0 -72
  131. package/dist-cjs/definitions/browser/pw-snapshot.js +0 -307
  132. package/dist-cjs/definitions/browser/pw-state.js +0 -70
  133. package/dist-cjs/definitions/browser/types.js +0 -5
  134. package/dist-cjs/definitions/code-intelligence.js +0 -473
  135. package/dist-cjs/definitions/core.js +0 -133
  136. package/dist-cjs/definitions/delegation.js +0 -515
  137. package/dist-cjs/definitions/deploy.js +0 -68
  138. package/dist-cjs/definitions/filesystem.js +0 -199
  139. package/dist-cjs/definitions/frg.js +0 -66
  140. package/dist-cjs/definitions/index.js +0 -43
  141. package/dist-cjs/definitions/memory.js +0 -126
  142. package/dist-cjs/definitions/messaging.js +0 -631
  143. package/dist-cjs/definitions/meta.js +0 -352
  144. package/dist-cjs/definitions/network.js +0 -162
  145. package/dist-cjs/definitions/outlook.js +0 -280
  146. package/dist-cjs/definitions/patch/apply-patch.js +0 -191
  147. package/dist-cjs/definitions/patch/fuzzy-match.js +0 -172
  148. package/dist-cjs/definitions/patch/index.js +0 -5
  149. package/dist-cjs/definitions/patch/patch-parser.js +0 -215
  150. package/dist-cjs/definitions/patch/sandbox-paths.js +0 -113
  151. package/dist-cjs/definitions/process/index.js +0 -8
  152. package/dist-cjs/definitions/process/process-registry.js +0 -231
  153. package/dist-cjs/definitions/process/process.js +0 -389
  154. package/dist-cjs/definitions/process/pty-keys.js +0 -259
  155. package/dist-cjs/definitions/process/session-slug.js +0 -145
  156. package/dist-cjs/definitions/quip.js +0 -198
  157. package/dist-cjs/definitions/search.js +0 -63
  158. package/dist-cjs/definitions/session-history.js +0 -72
  159. package/dist-cjs/definitions/shell.js +0 -184
  160. package/dist-cjs/definitions/slack.js +0 -183
  161. package/dist-cjs/definitions/web.js +0 -112
  162. package/dist-cjs/executors/apply-patch.js +0 -938
  163. package/dist-cjs/executors/arion.js +0 -125
  164. package/dist-cjs/executors/code-intelligence.js +0 -925
  165. package/dist-cjs/executors/deploy.js +0 -869
  166. package/dist-cjs/executors/filesystem.js +0 -1167
  167. package/dist-cjs/executors/frg-freshness.js +0 -627
  168. package/dist-cjs/executors/frg.js +0 -334
  169. package/dist-cjs/executors/index.js +0 -143
  170. package/dist-cjs/executors/learning-meta.js +0 -1165
  171. package/dist-cjs/executors/lsp-client.js +0 -310
  172. package/dist-cjs/executors/memory.js +0 -796
  173. package/dist-cjs/executors/meta.js +0 -226
  174. package/dist-cjs/executors/process-registry.js +0 -469
  175. package/dist-cjs/executors/pty-session-store.js +0 -34
  176. package/dist-cjs/executors/pty.js +0 -312
  177. package/dist-cjs/executors/restart.js +0 -155
  178. package/dist-cjs/executors/search-freshness.js +0 -234
  179. package/dist-cjs/executors/search-types.js +0 -56
  180. package/dist-cjs/executors/search.js +0 -102
  181. package/dist-cjs/executors/self-diagnose.js +0 -434
  182. package/dist-cjs/executors/session-history.js +0 -320
  183. package/dist-cjs/executors/shell-safety.js +0 -478
  184. package/dist-cjs/executors/shell.js +0 -1001
  185. package/dist-cjs/executors/utils.js +0 -73
  186. package/dist-cjs/executors/web.js +0 -547
  187. package/dist-cjs/extraction/content-extraction.js +0 -243
  188. package/dist-cjs/extraction/index.js +0 -8
  189. package/dist-cjs/headless-control-contract.js +0 -972
  190. package/dist-cjs/local-control-http-auth.js +0 -5
  191. package/dist-cjs/mcp/client.js +0 -185
  192. package/dist-cjs/mcp/connection.js +0 -484
  193. package/dist-cjs/mcp/index.js +0 -30
  194. package/dist-cjs/mcp/jsonrpc.js +0 -148
  195. package/dist-cjs/mcp/types.js +0 -8
  196. package/dist-cjs/network-control-adapter.js +0 -77
  197. package/dist-cjs/network-runtime/address-types.js +0 -168
  198. package/dist-cjs/network-runtime/db-owner-fencing.js +0 -76
  199. package/dist-cjs/network-runtime/delivery-receipts.js +0 -276
  200. package/dist-cjs/network-runtime/direct-endpoint-authority.js +0 -29
  201. package/dist-cjs/network-runtime/local-control-contract.js +0 -633
  202. package/dist-cjs/network-runtime/node-store-contract.js +0 -38
  203. package/dist-cjs/network-runtime/pair-route-contract.js +0 -80
  204. package/dist-cjs/network-runtime/peer-capabilities.js +0 -37
  205. package/dist-cjs/network-runtime/peer-principal-ref.js +0 -15
  206. package/dist-cjs/network-runtime/peer-state-machine.js +0 -129
  207. package/dist-cjs/network-runtime/protocol-schemas.js +0 -212
  208. package/dist-cjs/network-runtime/runtime-bootstrap-contract.js +0 -63
  209. package/dist-cjs/outlook/desktop-session.js +0 -318
  210. package/dist-cjs/policy.js +0 -155
  211. package/dist-cjs/providers/brave.js +0 -66
  212. package/dist-cjs/providers/duckduckgo.js +0 -180
  213. package/dist-cjs/providers/exa.js +0 -67
  214. package/dist-cjs/providers/firecrawl.js +0 -59
  215. package/dist-cjs/providers/index.js +0 -17
  216. package/dist-cjs/providers/jina.js +0 -53
  217. package/dist-cjs/providers/router.js +0 -100
  218. package/dist-cjs/providers/search-provider.js +0 -36
  219. package/dist-cjs/providers/tavily.js +0 -58
  220. package/dist-cjs/quip/desktop-session.js +0 -353
  221. package/dist-cjs/registry/index.js +0 -6
  222. package/dist-cjs/registry/registry.js +0 -761
  223. package/dist-cjs/runtime-socket-local-control-client.js +0 -367
  224. package/dist-cjs/security/dns-normalization.js +0 -22
  225. package/dist-cjs/security/dns-pinning.js +0 -160
  226. package/dist-cjs/security/external-content.js +0 -95
  227. package/dist-cjs/security/ssrf.js +0 -221
  228. package/dist-cjs/slack/desktop-session.js +0 -366
  229. package/dist-cjs/tool-factory.js +0 -50
  230. package/dist-cjs/types.js +0 -8
  231. package/dist-cjs/utils/retry.js +0 -169
  232. package/dist-cjs/utils/safe-parse-json.js +0 -164
  233. package/dist-cjs/utils/url.js +0 -23
@@ -1,398 +0,0 @@
1
- /**
2
- * @aria/tools - Self-diagnose tool executor
3
- *
4
- * Pure signal diagnostic tool. 5 sections, zero noise:
5
- * 1. runtime — process health + resource leaks
6
- * 2. errors — structured errors from JSONL (active + rotated)
7
- * 3. crashes — crash markers + dumps
8
- * 4. databases — sizes, WAL health, schema version
9
- * 5. daemon — daemon audit + delegation journal
10
- */
11
- import * as fs from "node:fs/promises";
12
- import * as path from "node:path";
13
- import * as os from "node:os";
14
- import { success, fail } from "./utils.js";
15
- // ============================================================================
16
- // Constants
17
- // ============================================================================
18
- const ALL_SECTIONS = ["runtime", "errors", "crashes", "databases", "daemon", "network"];
19
- const ARIA_HOME = path.join(os.homedir(), ".aria");
20
- // ============================================================================
21
- // Helpers
22
- // ============================================================================
23
- async function safeStat(p) {
24
- try {
25
- const s = await fs.stat(p);
26
- return { size: s.size, mtime: s.mtime };
27
- }
28
- catch {
29
- return null;
30
- }
31
- }
32
- function truncate(s, max) {
33
- if (!s)
34
- return "";
35
- return s.length <= max ? s : s.slice(0, max - 3) + "...";
36
- }
37
- async function dirCount(dir) {
38
- try {
39
- const entries = await fs.readdir(dir);
40
- return entries.length;
41
- }
42
- catch {
43
- return 0;
44
- }
45
- }
46
- async function tailJsonl(filePath, maxLines) {
47
- try {
48
- const content = await fs.readFile(filePath, "utf-8");
49
- const lines = content.trim().split("\n").slice(-maxLines);
50
- return lines
51
- .map((l) => {
52
- try {
53
- return JSON.parse(l);
54
- }
55
- catch {
56
- return null;
57
- }
58
- })
59
- .filter(Boolean);
60
- }
61
- catch {
62
- return [];
63
- }
64
- }
65
- // ============================================================================
66
- // Section: runtime
67
- // ============================================================================
68
- async function getRuntime(ctx) {
69
- const mem = process.memoryUsage();
70
- // Resource leak detection
71
- const socketDirs = await dirCount(path.join(ARIA_HOME, "sock"));
72
- const ownerFiles = await dirCount(path.join(ARIA_HOME, "run/owners"));
73
- // Stuck restart detection
74
- let relaunchPending = false;
75
- try {
76
- const entries = await fs.readdir(path.join(ARIA_HOME, "relaunch-pending"));
77
- relaunchPending = entries.length > 0;
78
- }
79
- catch {
80
- /* dir may not exist */
81
- }
82
- return {
83
- pid: process.pid,
84
- uptime_s: Math.round(process.uptime()),
85
- heap_mb: Math.round(mem.heapUsed / 1e6),
86
- rss_mb: Math.round(mem.rss / 1e6),
87
- node_version: process.version,
88
- platform: process.platform,
89
- arch: process.arch,
90
- cwd: process.cwd(),
91
- session_id: ctx.currentSessionId ?? null,
92
- socket_dirs: socketDirs,
93
- owner_files: ownerFiles,
94
- relaunch_pending: relaunchPending,
95
- };
96
- }
97
- // ============================================================================
98
- // Section: errors (active + rotated JSONL, merged, filtered)
99
- // ============================================================================
100
- async function getErrors(since, focus, limit) {
101
- // Read both active and rotated logs
102
- const active = path.join(ARIA_HOME, "error-events.jsonl");
103
- const rotated = path.join(ARIA_HOME, "error-events.jsonl.1");
104
- const [activeEntries, rotatedEntries] = await Promise.all([
105
- tailJsonl(active, 200),
106
- tailJsonl(rotated, 100),
107
- ]);
108
- const all = [
109
- ...rotatedEntries.map((e) => ({ ...e, _src: "rotated" })),
110
- ...activeEntries.map((e) => ({ ...e, _src: "active" })),
111
- ];
112
- const filtered = all.filter((entry) => {
113
- const ts = entry.timestamp;
114
- if (ts) {
115
- const d = new Date(ts);
116
- if (d < since)
117
- return false;
118
- }
119
- if (focus) {
120
- const haystack = [
121
- String(entry.message ?? ""),
122
- String(entry.category ?? ""),
123
- String(entry.stackTrace ?? ""),
124
- ]
125
- .join(" ")
126
- .toLowerCase();
127
- if (!haystack.includes(focus))
128
- return false;
129
- }
130
- return true;
131
- });
132
- return filtered.slice(-limit).map((entry) => ({
133
- timestamp: String(entry.timestamp ?? ""),
134
- severity: String(entry.severity ?? "unknown"),
135
- category: String(entry.category ?? ""),
136
- message: String(entry.message ?? ""),
137
- stackTrace: truncate(String(entry.stackTrace ?? ""), 500),
138
- source: String(entry._src),
139
- }));
140
- }
141
- // ============================================================================
142
- // Section: crashes
143
- // ============================================================================
144
- async function getCrashes(limit) {
145
- const results = [];
146
- for (const subdir of ["crash-markers", "crash-dumps"]) {
147
- const dir = path.join(ARIA_HOME, subdir);
148
- try {
149
- const files = await fs.readdir(dir);
150
- const jsonFiles = files.filter((f) => f.endsWith(".json"));
151
- const stats = await Promise.all(jsonFiles.map(async (f) => {
152
- const p = path.join(dir, f);
153
- const s = await safeStat(p);
154
- return { file: f, path: p, stat: s };
155
- }));
156
- const sorted = stats
157
- .filter((f) => f.stat)
158
- .sort((a, b) => b.stat.mtime.getTime() - a.stat.mtime.getTime());
159
- for (const { file, path: fp, stat } of sorted.slice(0, limit)) {
160
- try {
161
- const raw = await fs.readFile(fp, "utf-8");
162
- const parsed = JSON.parse(raw);
163
- if (subdir === "crash-markers") {
164
- results.push({
165
- file: `${subdir}/${file}`,
166
- timestamp: stat.mtime.toISOString(),
167
- content: parsed,
168
- });
169
- }
170
- else {
171
- const msg = parsed.error_message ?? parsed.message ?? parsed.error ?? "";
172
- results.push({
173
- file: `${subdir}/${file}`,
174
- size: stat.size,
175
- timestamp: stat.mtime.toISOString(),
176
- error_message: truncate(String(msg), 200),
177
- });
178
- }
179
- }
180
- catch {
181
- results.push({
182
- file: `${subdir}/${file}`,
183
- timestamp: stat.mtime.toISOString(),
184
- error_message: "Failed to parse",
185
- });
186
- }
187
- }
188
- }
189
- catch {
190
- /* dir may not exist */
191
- }
192
- }
193
- return results;
194
- }
195
- // ============================================================================
196
- // Section: databases (sizes, WAL health, schema version)
197
- // ============================================================================
198
- async function getDatabases() {
199
- const results = [];
200
- // Key database paths (only real ones, not empty legacy ghosts)
201
- const dbPaths = [
202
- {
203
- name: "memory.db (main)",
204
- path: path.join(ARIA_HOME, "arions/ARIA/memory.db"),
205
- checkSchema: true,
206
- },
207
- { name: "history.db", path: path.join(ARIA_HOME, "history.db") },
208
- { name: "investigation-metrics.db", path: path.join(ARIA_HOME, "investigation-metrics.db") },
209
- { name: "network/state.db", path: path.join(ARIA_HOME, "network/state.db") },
210
- { name: "node/node-state.db", path: path.join(ARIA_HOME, "node/node-state.db") },
211
- ];
212
- for (const db of dbPaths) {
213
- const stat = await safeStat(db.path);
214
- if (!stat || stat.size === 0)
215
- continue;
216
- // WAL size — a large WAL means checkpoint isn't running
217
- const walStat = await safeStat(db.path + "-wal");
218
- const walMb = walStat && walStat.size > 0 ? (walStat.size / 1e6).toFixed(1) : null;
219
- // Schema version for the main DB (lightweight: only reads one int)
220
- let schemaVersion;
221
- if (db.checkSchema) {
222
- try {
223
- // Use a child process to avoid importing better-sqlite3 / locking the DB
224
- const { execFileSync } = await import("node:child_process");
225
- const out = execFileSync("sqlite3", [db.path, "SELECT MAX(version) FROM schema_version;"], {
226
- timeout: 1000,
227
- encoding: "utf-8",
228
- }).trim();
229
- schemaVersion = parseInt(out, 10) || undefined;
230
- }
231
- catch {
232
- // sqlite3 CLI not available or query failed — not critical
233
- }
234
- }
235
- const info = {
236
- name: db.name,
237
- size_mb: (stat.size / 1e6).toFixed(1),
238
- wal_mb: walMb,
239
- modified: stat.mtime.toISOString(),
240
- };
241
- if (schemaVersion !== undefined)
242
- info.schema_version = schemaVersion;
243
- results.push(info);
244
- }
245
- return results;
246
- }
247
- // ============================================================================
248
- // Section: daemon (audit + delegation journal)
249
- // ============================================================================
250
- async function getDaemon(limit) {
251
- const results = [];
252
- // Daemon audit log — restart loops, crashes, startup events
253
- const auditPath = path.join(ARIA_HOME, "arions/ARIA/daemon/audit.jsonl");
254
- const auditEntries = await tailJsonl(auditPath, limit);
255
- if (auditEntries.length > 0) {
256
- results.push({ source: "daemon/audit", entries: auditEntries });
257
- }
258
- // Delegation journal — worker spawn/complete/fail
259
- const delegationPath = path.join(ARIA_HOME, "arions/ARIA/delegation-journal.jsonl");
260
- const delegationEntries = await tailJsonl(delegationPath, limit);
261
- if (delegationEntries.length > 0) {
262
- results.push({ source: "delegation-journal", entries: delegationEntries });
263
- }
264
- return results;
265
- }
266
- // ============================================================================
267
- // Section: network (identity, peers, tunnels, TLS, diagnostics)
268
- // ============================================================================
269
- async function getNetwork(limit) {
270
- // Node identity from network/config.json
271
- let nodeId = null;
272
- let displayName = null;
273
- let listenPort = null;
274
- let externalEndpoint = null;
275
- try {
276
- const raw = await fs.readFile(path.join(ARIA_HOME, "network/config.json"), "utf-8");
277
- const cfg = JSON.parse(raw);
278
- nodeId = cfg.nodeId ?? null;
279
- displayName = cfg.localDisplayNameSnapshot ?? null;
280
- listenPort = cfg.listenPort ?? null;
281
- if (cfg.externalEndpoint) {
282
- externalEndpoint = `${cfg.externalEndpoint.address}:${cfg.externalEndpoint.port}`;
283
- }
284
- // Never expose private keys — only the fact we have them
285
- }
286
- catch {
287
- /* config may not exist */
288
- }
289
- // Peers from network/state.db (using sqlite3 CLI to avoid locking)
290
- const peers = [];
291
- let peerCount = 0;
292
- try {
293
- const { execFileSync } = await import("node:child_process");
294
- const out = execFileSync("sqlite3", [
295
- path.join(ARIA_HOME, "network/state.db"),
296
- "-json",
297
- `SELECT nodeId, displayName, status, lastSeen FROM network_peers ORDER BY lastSeen DESC LIMIT ${limit};`,
298
- ], { timeout: 1000, encoding: "utf-8" }).trim();
299
- if (out) {
300
- const rows = JSON.parse(out);
301
- for (const row of rows) {
302
- peers.push({
303
- node_id: row.nodeId ?? "",
304
- name: row.displayName ?? "",
305
- status: row.status ?? "unknown",
306
- last_seen: row.lastSeen ? new Date(row.lastSeen).toISOString() : null,
307
- });
308
- }
309
- }
310
- const countOut = execFileSync("sqlite3", [path.join(ARIA_HOME, "network/state.db"), "SELECT COUNT(*) FROM network_peers;"], { timeout: 1000, encoding: "utf-8" }).trim();
311
- peerCount = parseInt(countOut, 10) || 0;
312
- }
313
- catch {
314
- /* sqlite3 CLI not available or query failed */
315
- }
316
- // TLS state
317
- let tlsCertsPresent = false;
318
- let trustedCas = 0;
319
- try {
320
- await fs.access(path.join(ARIA_HOME, "network/tls/server.pem"));
321
- tlsCertsPresent = true;
322
- }
323
- catch {
324
- /* no TLS certs */
325
- }
326
- try {
327
- const cas = await fs.readdir(path.join(ARIA_HOME, "network/trusted-cas"));
328
- trustedCas = cas.filter((f) => f.endsWith(".pem")).length;
329
- }
330
- catch {
331
- /* dir may not exist */
332
- }
333
- // Recent tunnel events (connection attempts, handshakes, failures)
334
- const tunnelEvents = await tailJsonl(path.join(ARIA_HOME, "audit/wireguard-secure-tunnel.jsonl"), limit);
335
- // Recent diagnostics (disconnects, reconnects, errors)
336
- const diagnostics = await tailJsonl(path.join(ARIA_HOME, "audit/wireguard-diagnostics.jsonl"), limit);
337
- return {
338
- node_id: nodeId,
339
- display_name: displayName,
340
- listen_port: listenPort,
341
- external_endpoint: externalEndpoint,
342
- peer_count: peerCount,
343
- peers,
344
- tls_certs_present: tlsCertsPresent,
345
- trusted_cas: trustedCas,
346
- recent_tunnel_events: tunnelEvents,
347
- recent_diagnostics: diagnostics,
348
- };
349
- }
350
- // ============================================================================
351
- // Main Executor
352
- // ============================================================================
353
- export async function executeSelfDiagnose(input, ctx) {
354
- if (ctx.abortSignal?.aborted)
355
- return fail("Operation cancelled");
356
- const startMs = Date.now();
357
- const limit = input.limit ?? 10;
358
- const since = input.since ? new Date(input.since) : new Date(Date.now() - 3600_000);
359
- const focus = input.focus?.toLowerCase();
360
- const requested = input.sections ?? ALL_SECTIONS;
361
- const sectionsToRun = requested.filter((s) => ALL_SECTIONS.includes(s));
362
- const sectionFns = {
363
- runtime: () => getRuntime(ctx),
364
- errors: () => getErrors(since, focus, limit),
365
- crashes: () => getCrashes(limit),
366
- databases: () => getDatabases(),
367
- daemon: () => getDaemon(limit),
368
- network: () => getNetwork(limit),
369
- };
370
- const results = {};
371
- const errors = [];
372
- const returned = [];
373
- const withTimeout = (fn, name) => Promise.race([
374
- fn().then((r) => {
375
- returned.push(name);
376
- return r;
377
- }),
378
- new Promise((_, reject) => setTimeout(() => reject(new Error(`${name} timed out`)), 2000)),
379
- ]).catch((err) => {
380
- errors.push(`${name}: ${err.message}`);
381
- return null;
382
- });
383
- await Promise.all(sectionsToRun.map(async (name) => {
384
- const fn = sectionFns[name];
385
- if (fn)
386
- results[name] = await withTimeout(fn, name);
387
- }));
388
- const elapsed = Date.now() - startMs;
389
- results._meta = {
390
- sections_requested: sectionsToRun,
391
- sections_returned: returned,
392
- elapsed_ms: elapsed,
393
- errors,
394
- };
395
- const msg = `Diagnostics: ${returned.length}/${sectionsToRun.length} sections in ${elapsed}ms` +
396
- (errors.length > 0 ? ` (${errors.length} errors)` : "");
397
- return success(msg, results);
398
- }
@@ -1,283 +0,0 @@
1
- /**
2
- * @aria/tools - Session History tool executor
3
- *
4
- * Implementation of session history operations for ARIA tool system.
5
- * Provides list, search, get, current, stats, delete, and set_title actions
6
- * against the SessionHistory SQLite database.
7
- */
8
- import * as os from "node:os";
9
- import * as path from "node:path";
10
- import { success, fail, getErrorMessage } from "./utils.js";
11
- // ============================================================================
12
- // Helpers
13
- // ============================================================================
14
- /** Max content length per message in 'get' output to prevent context blowout */
15
- const MAX_MESSAGE_CONTENT_LENGTH = 2000;
16
- /**
17
- * Cached fallback SessionHistory instance — created once per process via
18
- * dynamic import of @aria/cli's SessionHistory class. This avoids a hard
19
- * package dependency (tools → cli) while still giving the tool access to
20
- * the real SQLite history DB when ctx.sessionHistory isn't wired through.
21
- */
22
- /** Cached fallback SessionHistory — resolved once per process via dynamic import. */
23
- let _fallbackSessionHistory;
24
- let _fallbackPromise = null;
25
- async function resolveFallbackSessionHistory() {
26
- if (_fallbackSessionHistory !== undefined)
27
- return _fallbackSessionHistory;
28
- if (_fallbackPromise)
29
- return _fallbackPromise;
30
- _fallbackPromise = (async () => {
31
- try {
32
- // Dynamic import of @aria/cli (optionalDependency, ESM-only).
33
- // Using a variable prevents TS from resolving the specifier statically
34
- // and from downleveling to require() in CJS output.
35
- const _specifier = "@aria-cli/cli";
36
- const mod = (await import(_specifier));
37
- const SessionHistory = mod.SessionHistory;
38
- const ariaHome = process.env.ARIA_HOME ?? path.join(os.homedir(), ".aria");
39
- // Determine active arion from config or default to "ARIA"
40
- let arionName = "ARIA";
41
- try {
42
- const { readFileSync, existsSync } = await import("node:fs");
43
- const configPath = path.join(ariaHome, "config.json");
44
- if (existsSync(configPath)) {
45
- const raw = JSON.parse(readFileSync(configPath, "utf-8"));
46
- if (raw.activeArion)
47
- arionName = raw.activeArion;
48
- }
49
- }
50
- catch {
51
- // Ignore config read errors
52
- }
53
- const dbPath = SessionHistory.resolvePerArionPath(ariaHome, arionName);
54
- _fallbackSessionHistory = new SessionHistory(dbPath);
55
- return _fallbackSessionHistory;
56
- }
57
- catch {
58
- // @aria/cli not available (e.g. isolated worker) — no fallback
59
- _fallbackSessionHistory = null;
60
- return null;
61
- }
62
- })();
63
- return _fallbackPromise;
64
- }
65
- /**
66
- * Get the SessionHistoryRef — from ToolContext if wired, otherwise create
67
- * a fallback instance by dynamically importing @aria/cli's SessionHistory.
68
- */
69
- export async function getSessionHistory(ctx) {
70
- if (ctx.sessionHistory)
71
- return ctx.sessionHistory;
72
- return resolveFallbackSessionHistory();
73
- }
74
- function formatSessions(sessions) {
75
- return sessions.map((s) => ({
76
- id: s.id,
77
- title: s.title,
78
- arion: s.arion,
79
- model: s.model,
80
- messageCount: s.messageCount,
81
- updatedAt: s.updatedAt.toISOString(),
82
- completedAt: s.completedAt?.toISOString(),
83
- preview: s.preview,
84
- }));
85
- }
86
- function truncateContent(content, max) {
87
- if (content.length <= max)
88
- return content;
89
- return content.slice(0, max - 3) + "...";
90
- }
91
- function resolveSessionId(sh, rawId) {
92
- // Try exact match first (full UUID), then prefix
93
- if (rawId.length === 36) {
94
- const session = sh.getSession(rawId);
95
- return session ? rawId : null;
96
- }
97
- return sh.findSessionByPrefix(rawId);
98
- }
99
- // ============================================================================
100
- // Main Executor
101
- // ============================================================================
102
- /**
103
- * Execute a session_history action.
104
- */
105
- export async function executeSessionHistory(input, ctx) {
106
- if (ctx.abortSignal?.aborted)
107
- return fail("Operation cancelled");
108
- if (!input.action) {
109
- return fail("action is required. Valid actions: list, search, get, current, stats, delete, set_title");
110
- }
111
- const sh = await getSessionHistory(ctx);
112
- if (!sh) {
113
- return fail("Session history is not available in this context");
114
- }
115
- try {
116
- switch (input.action) {
117
- case "list":
118
- return executeList(input, sh);
119
- case "search":
120
- return executeSearch(input, sh);
121
- case "get":
122
- return executeGet(input, sh);
123
- case "current":
124
- return executeCurrent(ctx, sh);
125
- case "stats":
126
- return executeStats(sh);
127
- case "delete":
128
- return executeDelete(input, sh);
129
- case "set_title":
130
- return executeSetTitle(input, sh);
131
- default:
132
- return fail(`Unknown action "${input.action}". Valid actions: list, search, get, current, stats, delete, set_title`);
133
- }
134
- }
135
- catch (err) {
136
- return fail(getErrorMessage(err));
137
- }
138
- }
139
- // ============================================================================
140
- // Action Implementations
141
- // ============================================================================
142
- function executeList(input, sh) {
143
- const limit = input.limit ?? 20;
144
- const offset = input.offset ?? 0;
145
- const sessions = sh.listSessions(limit, offset);
146
- const formatted = formatSessions(sessions);
147
- const output = {
148
- sessions: formatted,
149
- total: formatted.length,
150
- limit,
151
- offset,
152
- };
153
- if (formatted.length === 0) {
154
- return success("No sessions found", output);
155
- }
156
- const summaryLines = formatted.map((s) => `• ${s.id.slice(0, 8)}… | ${s.title || "(untitled)"} | ${s.messageCount} msgs | ${s.arion} | ${s.updatedAt.slice(0, 10)}`);
157
- return success(`Found ${formatted.length} session(s):\n${summaryLines.join("\n")}`, output);
158
- }
159
- function executeSearch(input, sh) {
160
- if (!input.query || input.query.trim() === "") {
161
- return fail("query is required for 'search' action");
162
- }
163
- const limit = input.limit ?? 20;
164
- const offset = input.offset ?? 0;
165
- const sessions = sh.searchSessionsFts(input.query, limit, offset);
166
- const formatted = formatSessions(sessions);
167
- const output = {
168
- sessions: formatted,
169
- total: formatted.length,
170
- limit,
171
- offset,
172
- };
173
- if (formatted.length === 0) {
174
- return success(`No sessions found matching "${input.query}"`, output);
175
- }
176
- const summaryLines = formatted.map((s) => `• ${s.id.slice(0, 8)}… | ${s.title || "(untitled)"} | ${s.messageCount} msgs | ${s.arion} | ${s.updatedAt.slice(0, 10)}`);
177
- return success(`Found ${formatted.length} session(s) matching "${input.query}":\n${summaryLines.join("\n")}`, output);
178
- }
179
- function executeGet(input, sh) {
180
- if (!input.sessionId || input.sessionId.trim() === "") {
181
- return fail("sessionId is required for 'get' action");
182
- }
183
- const resolvedId = resolveSessionId(sh, input.sessionId);
184
- if (!resolvedId) {
185
- return fail(`Session not found for ID/prefix "${input.sessionId}"`);
186
- }
187
- const session = sh.getSession(resolvedId);
188
- if (!session) {
189
- return fail(`Session ${resolvedId} not found`);
190
- }
191
- const messageLimit = input.messageLimit ?? 50;
192
- // Take the last N messages to avoid context blowout
193
- const messagesSlice = session.messages.slice(-messageLimit);
194
- const formattedMessages = messagesSlice.map((m) => ({
195
- role: m.role,
196
- content: truncateContent(m.content, MAX_MESSAGE_CONTENT_LENGTH),
197
- createdAt: m.createdAt.toISOString(),
198
- toolCallId: m.toolCallId,
199
- }));
200
- // Also fetch metrics if available (getRunMetrics may not exist on all impls)
201
- let metrics;
202
- try {
203
- const rawMetrics = sh.getRunMetrics?.(resolvedId) ?? [];
204
- metrics =
205
- rawMetrics.length > 0
206
- ? rawMetrics.map((rm) => ({
207
- turnCount: rm.turnCount,
208
- inputTokens: rm.inputTokens,
209
- outputTokens: rm.outputTokens,
210
- estimatedCost: rm.estimatedCost,
211
- wallTimeMs: rm.wallTimeMs,
212
- }))
213
- : undefined;
214
- }
215
- catch {
216
- // Non-critical — metrics may not be available
217
- }
218
- const output = {
219
- id: session.id,
220
- arion: session.arion,
221
- model: session.model,
222
- messageCount: session.messages.length,
223
- messages: formattedMessages,
224
- metrics,
225
- };
226
- const truncNote = session.messages.length > messageLimit
227
- ? ` (showing last ${messageLimit} of ${session.messages.length})`
228
- : "";
229
- return success(`Session ${session.id.slice(0, 8)}… | ${session.arion} | ${session.model} | ${session.messages.length} messages${truncNote}`, output);
230
- }
231
- function executeCurrent(ctx, sh) {
232
- if (!ctx.currentSessionId) {
233
- return fail("No current session ID available");
234
- }
235
- const output = {
236
- sessionId: ctx.currentSessionId,
237
- };
238
- // Try to get session details if available
239
- const session = sh.getSession(ctx.currentSessionId);
240
- if (session) {
241
- return success(`Current session: ${session.id.slice(0, 8)}… | ${session.arion} | ${session.model} | ${session.messages.length} messages`, {
242
- ...output,
243
- arion: session.arion,
244
- model: session.model,
245
- messageCount: session.messages.length,
246
- });
247
- }
248
- return success(`Current session: ${ctx.currentSessionId}`, output);
249
- }
250
- function executeStats(sh) {
251
- const totalSessions = sh.getSessionCount?.() ?? 0;
252
- const incompleteSessions = (sh.getIncompleteSessions?.(1000) ?? []).length;
253
- const output = {
254
- totalSessions,
255
- incompleteSessions,
256
- };
257
- return success(`Session stats: ${totalSessions} total, ${incompleteSessions} incomplete, ${totalSessions - incompleteSessions} completed`, output);
258
- }
259
- function executeDelete(input, sh) {
260
- if (!input.sessionId || input.sessionId.trim() === "") {
261
- return fail("sessionId is required for 'delete' action");
262
- }
263
- const resolvedId = resolveSessionId(sh, input.sessionId);
264
- if (!resolvedId) {
265
- return fail(`Session not found for ID/prefix "${input.sessionId}"`);
266
- }
267
- sh.deleteSession(resolvedId);
268
- return success(`Deleted session ${resolvedId}`);
269
- }
270
- function executeSetTitle(input, sh) {
271
- if (!input.sessionId || input.sessionId.trim() === "") {
272
- return fail("sessionId is required for 'set_title' action");
273
- }
274
- if (!input.title || input.title.trim() === "") {
275
- return fail("title is required for 'set_title' action");
276
- }
277
- const resolvedId = resolveSessionId(sh, input.sessionId);
278
- if (!resolvedId) {
279
- return fail(`Session not found for ID/prefix "${input.sessionId}"`);
280
- }
281
- sh.setSessionTitle(resolvedId, input.title);
282
- return success(`Set title of session ${resolvedId.slice(0, 8)}… to "${input.title}"`);
283
- }