@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.
- package/dist/index.js +378 -70
- package/dist/network-runtime/index.js +8 -12
- package/dist-cjs/index.js +400 -435
- package/dist-cjs/network-runtime/index.js +8 -172
- package/package.json +8 -6
- package/dist/.tsbuildinfo +0 -1
- package/dist/ask-user-interaction.js +0 -22
- package/dist/cache/web-cache.js +0 -66
- package/dist/definitions/arion.js +0 -104
- package/dist/definitions/browser/browser.js +0 -418
- package/dist/definitions/browser/index.js +0 -4
- package/dist/definitions/browser/pw-downloads.js +0 -114
- package/dist/definitions/browser/pw-interactions.js +0 -199
- package/dist/definitions/browser/pw-responses.js +0 -76
- package/dist/definitions/browser/pw-session.js +0 -310
- package/dist/definitions/browser/pw-shared.js +0 -66
- package/dist/definitions/browser/pw-snapshot.js +0 -301
- package/dist/definitions/browser/pw-state.js +0 -62
- package/dist/definitions/browser/types.js +0 -4
- package/dist/definitions/code-intelligence.js +0 -470
- package/dist/definitions/core.js +0 -109
- package/dist/definitions/delegation.js +0 -512
- package/dist/definitions/deploy.js +0 -65
- package/dist/definitions/filesystem.js +0 -196
- package/dist/definitions/frg.js +0 -63
- package/dist/definitions/index.js +0 -20
- package/dist/definitions/memory.js +0 -123
- package/dist/definitions/messaging.js +0 -625
- package/dist/definitions/meta.js +0 -349
- package/dist/definitions/network.js +0 -159
- package/dist/definitions/outlook.js +0 -277
- package/dist/definitions/patch/apply-patch.js +0 -184
- package/dist/definitions/patch/fuzzy-match.js +0 -166
- package/dist/definitions/patch/index.js +0 -1
- package/dist/definitions/patch/patch-parser.js +0 -207
- package/dist/definitions/patch/sandbox-paths.js +0 -105
- package/dist/definitions/process/index.js +0 -4
- package/dist/definitions/process/process-registry.js +0 -213
- package/dist/definitions/process/process.js +0 -386
- package/dist/definitions/process/pty-keys.js +0 -254
- package/dist/definitions/process/session-slug.js +0 -142
- package/dist/definitions/quip.js +0 -195
- package/dist/definitions/search.js +0 -60
- package/dist/definitions/session-history.js +0 -69
- package/dist/definitions/shell.js +0 -181
- package/dist/definitions/slack.js +0 -180
- package/dist/definitions/web.js +0 -109
- package/dist/executors/apply-patch.js +0 -901
- package/dist/executors/arion.js +0 -119
- package/dist/executors/code-intelligence.js +0 -882
- package/dist/executors/deploy.js +0 -848
- package/dist/executors/filesystem.js +0 -1122
- package/dist/executors/frg-freshness.js +0 -576
- package/dist/executors/frg.js +0 -298
- package/dist/executors/index.js +0 -46
- package/dist/executors/learning-meta.js +0 -1146
- package/dist/executors/lsp-client.js +0 -296
- package/dist/executors/memory.js +0 -750
- package/dist/executors/meta.js +0 -220
- package/dist/executors/process-registry.js +0 -465
- package/dist/executors/pty-session-store.js +0 -30
- package/dist/executors/pty.js +0 -271
- package/dist/executors/restart.js +0 -119
- package/dist/executors/search-freshness.js +0 -195
- package/dist/executors/search-types.js +0 -52
- package/dist/executors/search.js +0 -66
- package/dist/executors/self-diagnose.js +0 -398
- package/dist/executors/session-history.js +0 -283
- package/dist/executors/shell-safety.js +0 -473
- package/dist/executors/shell.js +0 -954
- package/dist/executors/utils.js +0 -33
- package/dist/executors/web.js +0 -542
- package/dist/extraction/content-extraction.js +0 -235
- package/dist/extraction/index.js +0 -4
- package/dist/headless-control-contract.js +0 -967
- package/dist/local-control-http-auth.js +0 -2
- package/dist/mcp/client.js +0 -181
- package/dist/mcp/connection.js +0 -480
- package/dist/mcp/index.js +0 -10
- package/dist/mcp/jsonrpc.js +0 -144
- package/dist/mcp/types.js +0 -7
- package/dist/network-control-adapter.js +0 -72
- package/dist/network-runtime/address-types.js +0 -165
- package/dist/network-runtime/db-owner-fencing.js +0 -69
- package/dist/network-runtime/delivery-receipts.js +0 -267
- package/dist/network-runtime/direct-endpoint-authority.js +0 -25
- package/dist/network-runtime/local-control-contract.js +0 -627
- package/dist/network-runtime/node-store-contract.js +0 -34
- package/dist/network-runtime/pair-route-contract.js +0 -77
- package/dist/network-runtime/peer-capabilities.js +0 -28
- package/dist/network-runtime/peer-principal-ref.js +0 -12
- package/dist/network-runtime/peer-state-machine.js +0 -121
- package/dist/network-runtime/protocol-schemas.js +0 -205
- package/dist/network-runtime/runtime-bootstrap-contract.js +0 -60
- package/dist/outlook/desktop-session.js +0 -279
- package/dist/policy.js +0 -149
- package/dist/providers/brave.js +0 -62
- package/dist/providers/duckduckgo.js +0 -176
- package/dist/providers/exa.js +0 -63
- package/dist/providers/firecrawl.js +0 -55
- package/dist/providers/index.js +0 -7
- package/dist/providers/jina.js +0 -49
- package/dist/providers/router.js +0 -96
- package/dist/providers/search-provider.js +0 -32
- package/dist/providers/tavily.js +0 -54
- package/dist/quip/desktop-session.js +0 -317
- package/dist/registry/index.js +0 -1
- package/dist/registry/registry.js +0 -756
- package/dist/runtime-socket-local-control-client.js +0 -330
- package/dist/security/dns-normalization.js +0 -19
- package/dist/security/dns-pinning.js +0 -123
- package/dist/security/external-content.js +0 -91
- package/dist/security/ssrf.js +0 -181
- package/dist/slack/desktop-session.js +0 -324
- package/dist/tool-factory.js +0 -47
- package/dist/types.js +0 -7
- package/dist/utils/retry.js +0 -132
- package/dist/utils/safe-parse-json.js +0 -160
- package/dist/utils/url.js +0 -19
- package/dist-cjs/.tsbuildinfo +0 -1
- package/dist-cjs/ask-user-interaction.js +0 -27
- package/dist-cjs/cache/web-cache.js +0 -70
- package/dist-cjs/definitions/arion.js +0 -107
- package/dist-cjs/definitions/browser/browser.js +0 -421
- package/dist-cjs/definitions/browser/index.js +0 -8
- package/dist-cjs/definitions/browser/pw-downloads.js +0 -117
- package/dist-cjs/definitions/browser/pw-interactions.js +0 -213
- package/dist-cjs/definitions/browser/pw-responses.js +0 -84
- package/dist-cjs/definitions/browser/pw-session.js +0 -326
- package/dist-cjs/definitions/browser/pw-shared.js +0 -72
- package/dist-cjs/definitions/browser/pw-snapshot.js +0 -307
- package/dist-cjs/definitions/browser/pw-state.js +0 -70
- package/dist-cjs/definitions/browser/types.js +0 -5
- package/dist-cjs/definitions/code-intelligence.js +0 -473
- package/dist-cjs/definitions/core.js +0 -133
- package/dist-cjs/definitions/delegation.js +0 -515
- package/dist-cjs/definitions/deploy.js +0 -68
- package/dist-cjs/definitions/filesystem.js +0 -199
- package/dist-cjs/definitions/frg.js +0 -66
- package/dist-cjs/definitions/index.js +0 -43
- package/dist-cjs/definitions/memory.js +0 -126
- package/dist-cjs/definitions/messaging.js +0 -631
- package/dist-cjs/definitions/meta.js +0 -352
- package/dist-cjs/definitions/network.js +0 -162
- package/dist-cjs/definitions/outlook.js +0 -280
- package/dist-cjs/definitions/patch/apply-patch.js +0 -191
- package/dist-cjs/definitions/patch/fuzzy-match.js +0 -172
- package/dist-cjs/definitions/patch/index.js +0 -5
- package/dist-cjs/definitions/patch/patch-parser.js +0 -215
- package/dist-cjs/definitions/patch/sandbox-paths.js +0 -113
- package/dist-cjs/definitions/process/index.js +0 -8
- package/dist-cjs/definitions/process/process-registry.js +0 -231
- package/dist-cjs/definitions/process/process.js +0 -389
- package/dist-cjs/definitions/process/pty-keys.js +0 -259
- package/dist-cjs/definitions/process/session-slug.js +0 -145
- package/dist-cjs/definitions/quip.js +0 -198
- package/dist-cjs/definitions/search.js +0 -63
- package/dist-cjs/definitions/session-history.js +0 -72
- package/dist-cjs/definitions/shell.js +0 -184
- package/dist-cjs/definitions/slack.js +0 -183
- package/dist-cjs/definitions/web.js +0 -112
- package/dist-cjs/executors/apply-patch.js +0 -938
- package/dist-cjs/executors/arion.js +0 -125
- package/dist-cjs/executors/code-intelligence.js +0 -925
- package/dist-cjs/executors/deploy.js +0 -869
- package/dist-cjs/executors/filesystem.js +0 -1167
- package/dist-cjs/executors/frg-freshness.js +0 -627
- package/dist-cjs/executors/frg.js +0 -334
- package/dist-cjs/executors/index.js +0 -143
- package/dist-cjs/executors/learning-meta.js +0 -1165
- package/dist-cjs/executors/lsp-client.js +0 -310
- package/dist-cjs/executors/memory.js +0 -796
- package/dist-cjs/executors/meta.js +0 -226
- package/dist-cjs/executors/process-registry.js +0 -469
- package/dist-cjs/executors/pty-session-store.js +0 -34
- package/dist-cjs/executors/pty.js +0 -312
- package/dist-cjs/executors/restart.js +0 -155
- package/dist-cjs/executors/search-freshness.js +0 -234
- package/dist-cjs/executors/search-types.js +0 -56
- package/dist-cjs/executors/search.js +0 -102
- package/dist-cjs/executors/self-diagnose.js +0 -434
- package/dist-cjs/executors/session-history.js +0 -320
- package/dist-cjs/executors/shell-safety.js +0 -478
- package/dist-cjs/executors/shell.js +0 -1001
- package/dist-cjs/executors/utils.js +0 -73
- package/dist-cjs/executors/web.js +0 -547
- package/dist-cjs/extraction/content-extraction.js +0 -243
- package/dist-cjs/extraction/index.js +0 -8
- package/dist-cjs/headless-control-contract.js +0 -972
- package/dist-cjs/local-control-http-auth.js +0 -5
- package/dist-cjs/mcp/client.js +0 -185
- package/dist-cjs/mcp/connection.js +0 -484
- package/dist-cjs/mcp/index.js +0 -30
- package/dist-cjs/mcp/jsonrpc.js +0 -148
- package/dist-cjs/mcp/types.js +0 -8
- package/dist-cjs/network-control-adapter.js +0 -77
- package/dist-cjs/network-runtime/address-types.js +0 -168
- package/dist-cjs/network-runtime/db-owner-fencing.js +0 -76
- package/dist-cjs/network-runtime/delivery-receipts.js +0 -276
- package/dist-cjs/network-runtime/direct-endpoint-authority.js +0 -29
- package/dist-cjs/network-runtime/local-control-contract.js +0 -633
- package/dist-cjs/network-runtime/node-store-contract.js +0 -38
- package/dist-cjs/network-runtime/pair-route-contract.js +0 -80
- package/dist-cjs/network-runtime/peer-capabilities.js +0 -37
- package/dist-cjs/network-runtime/peer-principal-ref.js +0 -15
- package/dist-cjs/network-runtime/peer-state-machine.js +0 -129
- package/dist-cjs/network-runtime/protocol-schemas.js +0 -212
- package/dist-cjs/network-runtime/runtime-bootstrap-contract.js +0 -63
- package/dist-cjs/outlook/desktop-session.js +0 -318
- package/dist-cjs/policy.js +0 -155
- package/dist-cjs/providers/brave.js +0 -66
- package/dist-cjs/providers/duckduckgo.js +0 -180
- package/dist-cjs/providers/exa.js +0 -67
- package/dist-cjs/providers/firecrawl.js +0 -59
- package/dist-cjs/providers/index.js +0 -17
- package/dist-cjs/providers/jina.js +0 -53
- package/dist-cjs/providers/router.js +0 -100
- package/dist-cjs/providers/search-provider.js +0 -36
- package/dist-cjs/providers/tavily.js +0 -58
- package/dist-cjs/quip/desktop-session.js +0 -353
- package/dist-cjs/registry/index.js +0 -6
- package/dist-cjs/registry/registry.js +0 -761
- package/dist-cjs/runtime-socket-local-control-client.js +0 -367
- package/dist-cjs/security/dns-normalization.js +0 -22
- package/dist-cjs/security/dns-pinning.js +0 -160
- package/dist-cjs/security/external-content.js +0 -95
- package/dist-cjs/security/ssrf.js +0 -221
- package/dist-cjs/slack/desktop-session.js +0 -366
- package/dist-cjs/tool-factory.js +0 -50
- package/dist-cjs/types.js +0 -8
- package/dist-cjs/utils/retry.js +0 -169
- package/dist-cjs/utils/safe-parse-json.js +0 -164
- 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
|
-
}
|