loopsy 1.0.0
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/LICENSE +190 -0
- package/README.md +425 -0
- package/dist/cli/commands/connect.d.ts +2 -0
- package/dist/cli/commands/connect.d.ts.map +1 -0
- package/dist/cli/commands/connect.js +120 -0
- package/dist/cli/commands/connect.js.map +1 -0
- package/dist/cli/commands/context.d.ts +2 -0
- package/dist/cli/commands/context.d.ts.map +1 -0
- package/dist/cli/commands/context.js +39 -0
- package/dist/cli/commands/context.js.map +1 -0
- package/dist/cli/commands/daemon.d.ts +4 -0
- package/dist/cli/commands/daemon.d.ts.map +1 -0
- package/dist/cli/commands/daemon.js +55 -0
- package/dist/cli/commands/daemon.js.map +1 -0
- package/dist/cli/commands/dashboard.d.ts +2 -0
- package/dist/cli/commands/dashboard.d.ts.map +1 -0
- package/dist/cli/commands/dashboard.js +24 -0
- package/dist/cli/commands/dashboard.js.map +1 -0
- package/dist/cli/commands/doctor.d.ts +2 -0
- package/dist/cli/commands/doctor.d.ts.map +1 -0
- package/dist/cli/commands/doctor.js +130 -0
- package/dist/cli/commands/doctor.js.map +1 -0
- package/dist/cli/commands/exec.d.ts +2 -0
- package/dist/cli/commands/exec.d.ts.map +1 -0
- package/dist/cli/commands/exec.js +34 -0
- package/dist/cli/commands/exec.js.map +1 -0
- package/dist/cli/commands/init.d.ts +2 -0
- package/dist/cli/commands/init.d.ts.map +1 -0
- package/dist/cli/commands/init.js +71 -0
- package/dist/cli/commands/init.js.map +1 -0
- package/dist/cli/commands/key.d.ts +2 -0
- package/dist/cli/commands/key.d.ts.map +1 -0
- package/dist/cli/commands/key.js +39 -0
- package/dist/cli/commands/key.js.map +1 -0
- package/dist/cli/commands/logs.d.ts +2 -0
- package/dist/cli/commands/logs.d.ts.map +1 -0
- package/dist/cli/commands/logs.js +26 -0
- package/dist/cli/commands/logs.js.map +1 -0
- package/dist/cli/commands/mcp.d.ts +4 -0
- package/dist/cli/commands/mcp.d.ts.map +1 -0
- package/dist/cli/commands/mcp.js +70 -0
- package/dist/cli/commands/mcp.js.map +1 -0
- package/dist/cli/commands/pair.d.ts +6 -0
- package/dist/cli/commands/pair.d.ts.map +1 -0
- package/dist/cli/commands/pair.js +208 -0
- package/dist/cli/commands/pair.js.map +1 -0
- package/dist/cli/commands/peers.d.ts +2 -0
- package/dist/cli/commands/peers.d.ts.map +1 -0
- package/dist/cli/commands/peers.js +29 -0
- package/dist/cli/commands/peers.js.map +1 -0
- package/dist/cli/commands/service/linux.d.ts +7 -0
- package/dist/cli/commands/service/linux.d.ts.map +1 -0
- package/dist/cli/commands/service/linux.js +86 -0
- package/dist/cli/commands/service/linux.js.map +1 -0
- package/dist/cli/commands/service/macos.d.ts +7 -0
- package/dist/cli/commands/service/macos.d.ts.map +1 -0
- package/dist/cli/commands/service/macos.js +83 -0
- package/dist/cli/commands/service/macos.js.map +1 -0
- package/dist/cli/commands/service/windows.d.ts +7 -0
- package/dist/cli/commands/service/windows.d.ts.map +1 -0
- package/dist/cli/commands/service/windows.js +52 -0
- package/dist/cli/commands/service/windows.js.map +1 -0
- package/dist/cli/commands/service.d.ts +4 -0
- package/dist/cli/commands/service.d.ts.map +1 -0
- package/dist/cli/commands/service.js +68 -0
- package/dist/cli/commands/service.js.map +1 -0
- package/dist/cli/commands/session.d.ts +8 -0
- package/dist/cli/commands/session.d.ts.map +1 -0
- package/dist/cli/commands/session.js +270 -0
- package/dist/cli/commands/session.js.map +1 -0
- package/dist/cli/commands/transfer.d.ts +3 -0
- package/dist/cli/commands/transfer.d.ts.map +1 -0
- package/dist/cli/commands/transfer.js +57 -0
- package/dist/cli/commands/transfer.js.map +1 -0
- package/dist/cli/index.d.ts +3 -0
- package/dist/cli/index.d.ts.map +1 -0
- package/dist/cli/index.js +89 -0
- package/dist/cli/index.js.map +1 -0
- package/dist/cli/package-root.d.ts +27 -0
- package/dist/cli/package-root.d.ts.map +1 -0
- package/dist/cli/package-root.js +54 -0
- package/dist/cli/package-root.js.map +1 -0
- package/dist/cli/utils.d.ts +11 -0
- package/dist/cli/utils.d.ts.map +1 -0
- package/dist/cli/utils.js +48 -0
- package/dist/cli/utils.js.map +1 -0
- package/dist/daemon/config.d.ts +4 -0
- package/dist/daemon/config.d.ts.map +1 -0
- package/dist/daemon/config.js +58 -0
- package/dist/daemon/config.js.map +1 -0
- package/dist/daemon/hooks/permission-hook.mjs +108 -0
- package/dist/daemon/index.d.ts +3 -0
- package/dist/daemon/index.d.ts.map +1 -0
- package/dist/daemon/index.js +3 -0
- package/dist/daemon/index.js.map +1 -0
- package/dist/daemon/main.d.ts +3 -0
- package/dist/daemon/main.d.ts.map +1 -0
- package/dist/daemon/main.js +28 -0
- package/dist/daemon/main.js.map +1 -0
- package/dist/daemon/middleware/auth.d.ts +3 -0
- package/dist/daemon/middleware/auth.d.ts.map +1 -0
- package/dist/daemon/middleware/auth.js +22 -0
- package/dist/daemon/middleware/auth.js.map +1 -0
- package/dist/daemon/routes/ai-tasks.d.ts +4 -0
- package/dist/daemon/routes/ai-tasks.d.ts.map +1 -0
- package/dist/daemon/routes/ai-tasks.js +146 -0
- package/dist/daemon/routes/ai-tasks.js.map +1 -0
- package/dist/daemon/routes/context.d.ts +4 -0
- package/dist/daemon/routes/context.d.ts.map +1 -0
- package/dist/daemon/routes/context.js +49 -0
- package/dist/daemon/routes/context.js.map +1 -0
- package/dist/daemon/routes/execute.d.ts +4 -0
- package/dist/daemon/routes/execute.d.ts.map +1 -0
- package/dist/daemon/routes/execute.js +74 -0
- package/dist/daemon/routes/execute.js.map +1 -0
- package/dist/daemon/routes/health.d.ts +15 -0
- package/dist/daemon/routes/health.d.ts.map +1 -0
- package/dist/daemon/routes/health.js +38 -0
- package/dist/daemon/routes/health.js.map +1 -0
- package/dist/daemon/routes/pair.d.ts +11 -0
- package/dist/daemon/routes/pair.d.ts.map +1 -0
- package/dist/daemon/routes/pair.js +149 -0
- package/dist/daemon/routes/pair.js.map +1 -0
- package/dist/daemon/routes/peers.d.ts +5 -0
- package/dist/daemon/routes/peers.d.ts.map +1 -0
- package/dist/daemon/routes/peers.js +69 -0
- package/dist/daemon/routes/peers.js.map +1 -0
- package/dist/daemon/routes/transfer.d.ts +4 -0
- package/dist/daemon/routes/transfer.d.ts.map +1 -0
- package/dist/daemon/routes/transfer.js +135 -0
- package/dist/daemon/routes/transfer.js.map +1 -0
- package/dist/daemon/server.d.ts +8 -0
- package/dist/daemon/server.d.ts.map +1 -0
- package/dist/daemon/server.js +170 -0
- package/dist/daemon/server.js.map +1 -0
- package/dist/daemon/services/ai-task-manager.d.ts +56 -0
- package/dist/daemon/services/ai-task-manager.d.ts.map +1 -0
- package/dist/daemon/services/ai-task-manager.js +491 -0
- package/dist/daemon/services/ai-task-manager.js.map +1 -0
- package/dist/daemon/services/audit-logger.d.ts +16 -0
- package/dist/daemon/services/audit-logger.d.ts.map +1 -0
- package/dist/daemon/services/audit-logger.js +23 -0
- package/dist/daemon/services/audit-logger.js.map +1 -0
- package/dist/daemon/services/context-store.d.ts +17 -0
- package/dist/daemon/services/context-store.d.ts.map +1 -0
- package/dist/daemon/services/context-store.js +97 -0
- package/dist/daemon/services/context-store.js.map +1 -0
- package/dist/daemon/services/job-manager.d.ts +19 -0
- package/dist/daemon/services/job-manager.d.ts.map +1 -0
- package/dist/daemon/services/job-manager.js +92 -0
- package/dist/daemon/services/job-manager.js.map +1 -0
- package/dist/daemon/services/tls-manager.d.ts +33 -0
- package/dist/daemon/services/tls-manager.d.ts.map +1 -0
- package/dist/daemon/services/tls-manager.js +114 -0
- package/dist/daemon/services/tls-manager.js.map +1 -0
- package/dist/daemon/utils/which.d.ts +2 -0
- package/dist/daemon/utils/which.d.ts.map +1 -0
- package/dist/daemon/utils/which.js +18 -0
- package/dist/daemon/utils/which.js.map +1 -0
- package/dist/dashboard/config.d.ts +8 -0
- package/dist/dashboard/config.d.ts.map +1 -0
- package/dist/dashboard/config.js +22 -0
- package/dist/dashboard/config.js.map +1 -0
- package/dist/dashboard/public/app.js +120 -0
- package/dist/dashboard/public/icon-192.png +0 -0
- package/dist/dashboard/public/icon-512.png +0 -0
- package/dist/dashboard/public/index.html +85 -0
- package/dist/dashboard/public/manifest.json +12 -0
- package/dist/dashboard/public/style.css +784 -0
- package/dist/dashboard/public/sw.js +31 -0
- package/dist/dashboard/public/views/ai-tasks.js +679 -0
- package/dist/dashboard/public/views/context.js +167 -0
- package/dist/dashboard/public/views/messages.js +263 -0
- package/dist/dashboard/public/views/overview.js +228 -0
- package/dist/dashboard/public/views/peers.js +136 -0
- package/dist/dashboard/public/views/terminal.js +153 -0
- package/dist/dashboard/routes/ai-tasks.d.ts +3 -0
- package/dist/dashboard/routes/ai-tasks.d.ts.map +1 -0
- package/dist/dashboard/routes/ai-tasks.js +193 -0
- package/dist/dashboard/routes/ai-tasks.js.map +1 -0
- package/dist/dashboard/routes/messages.d.ts +3 -0
- package/dist/dashboard/routes/messages.d.ts.map +1 -0
- package/dist/dashboard/routes/messages.js +137 -0
- package/dist/dashboard/routes/messages.js.map +1 -0
- package/dist/dashboard/routes/peer-utils.d.ts +17 -0
- package/dist/dashboard/routes/peer-utils.d.ts.map +1 -0
- package/dist/dashboard/routes/peer-utils.js +193 -0
- package/dist/dashboard/routes/peer-utils.js.map +1 -0
- package/dist/dashboard/routes/peers-all.d.ts +3 -0
- package/dist/dashboard/routes/peers-all.d.ts.map +1 -0
- package/dist/dashboard/routes/peers-all.js +8 -0
- package/dist/dashboard/routes/peers-all.js.map +1 -0
- package/dist/dashboard/routes/proxy.d.ts +3 -0
- package/dist/dashboard/routes/proxy.d.ts.map +1 -0
- package/dist/dashboard/routes/proxy.js +59 -0
- package/dist/dashboard/routes/proxy.js.map +1 -0
- package/dist/dashboard/routes/sessions.d.ts +3 -0
- package/dist/dashboard/routes/sessions.d.ts.map +1 -0
- package/dist/dashboard/routes/sessions.js +64 -0
- package/dist/dashboard/routes/sessions.js.map +1 -0
- package/dist/dashboard/routes/sse.d.ts +3 -0
- package/dist/dashboard/routes/sse.d.ts.map +1 -0
- package/dist/dashboard/routes/sse.js +49 -0
- package/dist/dashboard/routes/sse.js.map +1 -0
- package/dist/dashboard/routes/status.d.ts +3 -0
- package/dist/dashboard/routes/status.d.ts.map +1 -0
- package/dist/dashboard/routes/status.js +38 -0
- package/dist/dashboard/routes/status.js.map +1 -0
- package/dist/dashboard/server.d.ts +3 -0
- package/dist/dashboard/server.d.ts.map +1 -0
- package/dist/dashboard/server.js +77 -0
- package/dist/dashboard/server.js.map +1 -0
- package/dist/dashboard/session-manager.d.ts +17 -0
- package/dist/dashboard/session-manager.d.ts.map +1 -0
- package/dist/dashboard/session-manager.js +225 -0
- package/dist/dashboard/session-manager.js.map +1 -0
- package/dist/discovery/health-checker.d.ts +15 -0
- package/dist/discovery/health-checker.d.ts.map +1 -0
- package/dist/discovery/health-checker.js +47 -0
- package/dist/discovery/health-checker.js.map +1 -0
- package/dist/discovery/index.d.ts +4 -0
- package/dist/discovery/index.d.ts.map +1 -0
- package/dist/discovery/index.js +4 -0
- package/dist/discovery/index.js.map +1 -0
- package/dist/discovery/mdns.d.ts +21 -0
- package/dist/discovery/mdns.d.ts.map +1 -0
- package/dist/discovery/mdns.js +83 -0
- package/dist/discovery/mdns.js.map +1 -0
- package/dist/discovery/peer-registry.d.ts +18 -0
- package/dist/discovery/peer-registry.d.ts.map +1 -0
- package/dist/discovery/peer-registry.js +81 -0
- package/dist/discovery/peer-registry.js.map +1 -0
- package/dist/mcp-server/daemon-client.d.ts +69 -0
- package/dist/mcp-server/daemon-client.d.ts.map +1 -0
- package/dist/mcp-server/daemon-client.js +281 -0
- package/dist/mcp-server/daemon-client.js.map +1 -0
- package/dist/mcp-server/index.d.ts +3 -0
- package/dist/mcp-server/index.d.ts.map +1 -0
- package/dist/mcp-server/index.js +406 -0
- package/dist/mcp-server/index.js.map +1 -0
- package/dist/protocol/constants.d.ts +66 -0
- package/dist/protocol/constants.d.ts.map +1 -0
- package/dist/protocol/constants.js +66 -0
- package/dist/protocol/constants.js.map +1 -0
- package/dist/protocol/errors.d.ts +47 -0
- package/dist/protocol/errors.d.ts.map +1 -0
- package/dist/protocol/errors.js +62 -0
- package/dist/protocol/errors.js.map +1 -0
- package/dist/protocol/index.d.ts +5 -0
- package/dist/protocol/index.d.ts.map +1 -0
- package/dist/protocol/index.js +5 -0
- package/dist/protocol/index.js.map +1 -0
- package/dist/protocol/schemas.d.ts +209 -0
- package/dist/protocol/schemas.d.ts.map +1 -0
- package/dist/protocol/schemas.js +115 -0
- package/dist/protocol/schemas.js.map +1 -0
- package/dist/protocol/types.d.ts +302 -0
- package/dist/protocol/types.d.ts.map +1 -0
- package/dist/protocol/types.js +2 -0
- package/dist/protocol/types.js.map +1 -0
- package/package.json +50 -0
- package/scripts/postinstall.mjs +42 -0
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
import { readFile, writeFile, mkdir } from 'node:fs/promises';
|
|
2
|
+
import { join } from 'node:path';
|
|
3
|
+
import { homedir } from 'node:os';
|
|
4
|
+
import { CONFIG_DIR, MAX_CONTEXT_ENTRIES, MAX_CONTEXT_VALUE_SIZE } from '@loopsy/protocol';
|
|
5
|
+
import { LoopsyError, LoopsyErrorCode } from '@loopsy/protocol';
|
|
6
|
+
export class ContextStore {
|
|
7
|
+
entries = new Map();
|
|
8
|
+
expiryTimer = null;
|
|
9
|
+
contextFile;
|
|
10
|
+
constructor(dataDir) {
|
|
11
|
+
const dir = dataDir ?? join(homedir(), CONFIG_DIR);
|
|
12
|
+
this.contextFile = join(dir, 'context.json');
|
|
13
|
+
}
|
|
14
|
+
async load() {
|
|
15
|
+
try {
|
|
16
|
+
const data = await readFile(this.contextFile, 'utf-8');
|
|
17
|
+
const items = JSON.parse(data);
|
|
18
|
+
for (const entry of items) {
|
|
19
|
+
if (!entry.expiresAt || entry.expiresAt > Date.now()) {
|
|
20
|
+
this.entries.set(entry.key, entry);
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
catch {
|
|
25
|
+
// No context file yet
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
async save() {
|
|
29
|
+
const dir = join(this.contextFile, '..');
|
|
30
|
+
await mkdir(dir, { recursive: true });
|
|
31
|
+
await writeFile(this.contextFile, JSON.stringify(Array.from(this.entries.values()), null, 2));
|
|
32
|
+
}
|
|
33
|
+
startExpiryCheck() {
|
|
34
|
+
this.expiryTimer = setInterval(() => {
|
|
35
|
+
const now = Date.now();
|
|
36
|
+
for (const [key, entry] of this.entries) {
|
|
37
|
+
if (entry.expiresAt && entry.expiresAt <= now) {
|
|
38
|
+
this.entries.delete(key);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
}, 10_000);
|
|
42
|
+
}
|
|
43
|
+
stopExpiryCheck() {
|
|
44
|
+
if (this.expiryTimer) {
|
|
45
|
+
clearInterval(this.expiryTimer);
|
|
46
|
+
this.expiryTimer = null;
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
set(key, value, fromNodeId, ttl) {
|
|
50
|
+
if (value.length > MAX_CONTEXT_VALUE_SIZE) {
|
|
51
|
+
throw new LoopsyError(LoopsyErrorCode.CONTEXT_VALUE_TOO_LARGE, `Value exceeds max size of ${MAX_CONTEXT_VALUE_SIZE} bytes`);
|
|
52
|
+
}
|
|
53
|
+
if (this.entries.size >= MAX_CONTEXT_ENTRIES && !this.entries.has(key)) {
|
|
54
|
+
throw new LoopsyError(LoopsyErrorCode.CONTEXT_MAX_ENTRIES, `Max context entries (${MAX_CONTEXT_ENTRIES}) reached`);
|
|
55
|
+
}
|
|
56
|
+
const now = Date.now();
|
|
57
|
+
const entry = {
|
|
58
|
+
key,
|
|
59
|
+
value,
|
|
60
|
+
fromNodeId,
|
|
61
|
+
createdAt: this.entries.get(key)?.createdAt ?? now,
|
|
62
|
+
updatedAt: now,
|
|
63
|
+
ttl,
|
|
64
|
+
expiresAt: ttl ? now + ttl * 1000 : undefined,
|
|
65
|
+
};
|
|
66
|
+
this.entries.set(key, entry);
|
|
67
|
+
return entry;
|
|
68
|
+
}
|
|
69
|
+
get(key) {
|
|
70
|
+
const entry = this.entries.get(key);
|
|
71
|
+
if (entry?.expiresAt && entry.expiresAt <= Date.now()) {
|
|
72
|
+
this.entries.delete(key);
|
|
73
|
+
return undefined;
|
|
74
|
+
}
|
|
75
|
+
return entry;
|
|
76
|
+
}
|
|
77
|
+
delete(key) {
|
|
78
|
+
return this.entries.delete(key);
|
|
79
|
+
}
|
|
80
|
+
list(prefix) {
|
|
81
|
+
const now = Date.now();
|
|
82
|
+
const result = [];
|
|
83
|
+
for (const [key, entry] of this.entries) {
|
|
84
|
+
if (entry.expiresAt && entry.expiresAt <= now) {
|
|
85
|
+
this.entries.delete(key);
|
|
86
|
+
}
|
|
87
|
+
else if (!prefix || key.startsWith(prefix)) {
|
|
88
|
+
result.push(entry);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
return result;
|
|
92
|
+
}
|
|
93
|
+
get size() {
|
|
94
|
+
return this.entries.size;
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
//# sourceMappingURL=context-store.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"context-store.js","sourceRoot":"","sources":["../../src/services/context-store.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import type { ExecuteParams, ExecuteResult, JobInfo } from '@loopsy/protocol';
|
|
2
|
+
export declare class JobManager {
|
|
3
|
+
private jobs;
|
|
4
|
+
private maxConcurrent;
|
|
5
|
+
private denylist;
|
|
6
|
+
private allowlist?;
|
|
7
|
+
constructor(opts: {
|
|
8
|
+
maxConcurrent?: number;
|
|
9
|
+
denylist?: string[];
|
|
10
|
+
allowlist?: string[];
|
|
11
|
+
});
|
|
12
|
+
execute(params: ExecuteParams, fromNodeId: string): Promise<ExecuteResult>;
|
|
13
|
+
cancel(jobId: string): boolean;
|
|
14
|
+
getActiveJobs(): JobInfo[];
|
|
15
|
+
killAll(): void;
|
|
16
|
+
get activeCount(): number;
|
|
17
|
+
private validateCommand;
|
|
18
|
+
}
|
|
19
|
+
//# sourceMappingURL=job-manager.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"job-manager.d.ts","sourceRoot":"","sources":["../../src/services/job-manager.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,aAAa,EAAE,aAAa,EAAE,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAG9E,qBAAa,UAAU;IACrB,OAAO,CAAC,IAAI,CAA+D;IAC3E,OAAO,CAAC,aAAa,CAAS;IAC9B,OAAO,CAAC,QAAQ,CAAW;IAC3B,OAAO,CAAC,SAAS,CAAC,CAAW;gBAEjB,IAAI,EAAE;QAAE,aAAa,CAAC,EAAE,MAAM,CAAC;QAAC,QAAQ,CAAC,EAAE,MAAM,EAAE,CAAC;QAAC,SAAS,CAAC,EAAE,MAAM,EAAE,CAAA;KAAE;IAMjF,OAAO,CAAC,MAAM,EAAE,aAAa,EAAE,UAAU,EAAE,MAAM,GAAG,OAAO,CAAC,aAAa,CAAC;IAsDhF,MAAM,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO;IAY9B,aAAa,IAAI,OAAO,EAAE;IAI1B,OAAO,IAAI,IAAI;IAOf,IAAI,WAAW,IAAI,MAAM,CAExB;IAED,OAAO,CAAC,eAAe;CASxB"}
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
import { spawn } from 'node:child_process';
|
|
2
|
+
import { randomUUID } from 'node:crypto';
|
|
3
|
+
import { LoopsyError, LoopsyErrorCode, MAX_CONCURRENT_JOBS } from '@loopsy/protocol';
|
|
4
|
+
export class JobManager {
|
|
5
|
+
jobs = new Map();
|
|
6
|
+
maxConcurrent;
|
|
7
|
+
denylist;
|
|
8
|
+
allowlist;
|
|
9
|
+
constructor(opts) {
|
|
10
|
+
this.maxConcurrent = opts.maxConcurrent ?? MAX_CONCURRENT_JOBS;
|
|
11
|
+
this.denylist = opts.denylist ?? [];
|
|
12
|
+
this.allowlist = opts.allowlist;
|
|
13
|
+
}
|
|
14
|
+
async execute(params, fromNodeId) {
|
|
15
|
+
this.validateCommand(params.command);
|
|
16
|
+
if (this.jobs.size >= this.maxConcurrent) {
|
|
17
|
+
throw new LoopsyError(LoopsyErrorCode.EXEC_MAX_CONCURRENT, `Max concurrent jobs (${this.maxConcurrent}) reached`);
|
|
18
|
+
}
|
|
19
|
+
const jobId = randomUUID();
|
|
20
|
+
const startedAt = Date.now();
|
|
21
|
+
return new Promise((resolve, reject) => {
|
|
22
|
+
const proc = spawn(params.command, params.args ?? [], {
|
|
23
|
+
cwd: params.cwd,
|
|
24
|
+
env: params.env ? { ...process.env, ...params.env } : process.env,
|
|
25
|
+
shell: false,
|
|
26
|
+
timeout: params.timeout,
|
|
27
|
+
});
|
|
28
|
+
const info = {
|
|
29
|
+
jobId,
|
|
30
|
+
command: params.command,
|
|
31
|
+
args: params.args ?? [],
|
|
32
|
+
startedAt,
|
|
33
|
+
fromNodeId,
|
|
34
|
+
pid: proc.pid,
|
|
35
|
+
};
|
|
36
|
+
this.jobs.set(jobId, { process: proc, info });
|
|
37
|
+
let stdout = '';
|
|
38
|
+
let stderr = '';
|
|
39
|
+
proc.stdout?.on('data', (chunk) => { stdout += chunk.toString(); });
|
|
40
|
+
proc.stderr?.on('data', (chunk) => { stderr += chunk.toString(); });
|
|
41
|
+
proc.on('close', (exitCode, signal) => {
|
|
42
|
+
this.jobs.delete(jobId);
|
|
43
|
+
resolve({
|
|
44
|
+
jobId,
|
|
45
|
+
exitCode,
|
|
46
|
+
stdout,
|
|
47
|
+
stderr,
|
|
48
|
+
duration: Date.now() - startedAt,
|
|
49
|
+
killed: signal !== null,
|
|
50
|
+
});
|
|
51
|
+
});
|
|
52
|
+
proc.on('error', (err) => {
|
|
53
|
+
this.jobs.delete(jobId);
|
|
54
|
+
reject(new LoopsyError(LoopsyErrorCode.EXEC_FAILED, `Execution failed: ${err.message}`));
|
|
55
|
+
});
|
|
56
|
+
});
|
|
57
|
+
}
|
|
58
|
+
cancel(jobId) {
|
|
59
|
+
const job = this.jobs.get(jobId);
|
|
60
|
+
if (!job)
|
|
61
|
+
return false;
|
|
62
|
+
job.process.kill('SIGTERM');
|
|
63
|
+
setTimeout(() => {
|
|
64
|
+
if (this.jobs.has(jobId)) {
|
|
65
|
+
job.process.kill('SIGKILL');
|
|
66
|
+
}
|
|
67
|
+
}, 5000);
|
|
68
|
+
return true;
|
|
69
|
+
}
|
|
70
|
+
getActiveJobs() {
|
|
71
|
+
return Array.from(this.jobs.values()).map((j) => j.info);
|
|
72
|
+
}
|
|
73
|
+
killAll() {
|
|
74
|
+
for (const [id, job] of this.jobs) {
|
|
75
|
+
job.process.kill('SIGKILL');
|
|
76
|
+
this.jobs.delete(id);
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
get activeCount() {
|
|
80
|
+
return this.jobs.size;
|
|
81
|
+
}
|
|
82
|
+
validateCommand(command) {
|
|
83
|
+
const base = command.split('/').pop() ?? command;
|
|
84
|
+
if (this.denylist.includes(base)) {
|
|
85
|
+
throw new LoopsyError(LoopsyErrorCode.EXEC_COMMAND_DENIED, `Command '${base}' is denied`);
|
|
86
|
+
}
|
|
87
|
+
if (this.allowlist && !this.allowlist.includes(base)) {
|
|
88
|
+
throw new LoopsyError(LoopsyErrorCode.EXEC_COMMAND_DENIED, `Command '${base}' is not in the allowlist`);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
//# sourceMappingURL=job-manager.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"job-manager.js","sourceRoot":"","sources":["../../src/services/job-manager.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import type { PeerCertInfo } from '@loopsy/protocol';
|
|
2
|
+
export interface TlsFiles {
|
|
3
|
+
cert: string;
|
|
4
|
+
key: string;
|
|
5
|
+
fingerprint: string;
|
|
6
|
+
}
|
|
7
|
+
export declare class TlsManager {
|
|
8
|
+
private dataDir;
|
|
9
|
+
private tlsDir;
|
|
10
|
+
private certPath;
|
|
11
|
+
private keyPath;
|
|
12
|
+
constructor(dataDir?: string);
|
|
13
|
+
/** Check if TLS cert/key already exist */
|
|
14
|
+
hasCerts(): boolean;
|
|
15
|
+
/** Generate a self-signed EC certificate if none exists */
|
|
16
|
+
ensureCerts(hostname: string): Promise<TlsFiles>;
|
|
17
|
+
/** Generate a new self-signed certificate */
|
|
18
|
+
generateCerts(hostname: string): Promise<TlsFiles>;
|
|
19
|
+
/** Load existing cert/key files */
|
|
20
|
+
loadCerts(): Promise<TlsFiles>;
|
|
21
|
+
/** Compute SHA-256 fingerprint of a PEM certificate */
|
|
22
|
+
computeFingerprint(certPem: string): string;
|
|
23
|
+
/** Get cert info for display */
|
|
24
|
+
getCertInfo(certPem: string): PeerCertInfo;
|
|
25
|
+
/** Get Fastify HTTPS options */
|
|
26
|
+
getHttpsOptions(): Promise<{
|
|
27
|
+
key: string;
|
|
28
|
+
cert: string;
|
|
29
|
+
} | null>;
|
|
30
|
+
/** Create a self-signed certificate using openssl CLI (widely available) */
|
|
31
|
+
private createSelfSignedCert;
|
|
32
|
+
}
|
|
33
|
+
//# sourceMappingURL=tls-manager.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tls-manager.d.ts","sourceRoot":"","sources":["../../src/services/tls-manager.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAAa,YAAY,EAAE,MAAM,kBAAkB,CAAC;AAMhE,MAAM,WAAW,QAAQ;IACvB,IAAI,EAAE,MAAM,CAAC;IACb,GAAG,EAAE,MAAM,CAAC;IACZ,WAAW,EAAE,MAAM,CAAC;CACrB;AAED,qBAAa,UAAU;IACrB,OAAO,CAAC,OAAO,CAAS;IACxB,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,QAAQ,CAAS;IACzB,OAAO,CAAC,OAAO,CAAS;gBAEZ,OAAO,CAAC,EAAE,MAAM;IAO5B,0CAA0C;IAC1C,QAAQ,IAAI,OAAO;IAInB,2DAA2D;IACrD,WAAW,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC;IAOtD,6CAA6C;IACvC,aAAa,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC;IA4BxD,mCAAmC;IAC7B,SAAS,IAAI,OAAO,CAAC,QAAQ,CAAC;IAOpC,uDAAuD;IACvD,kBAAkB,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM;IAU3C,gCAAgC;IAChC,WAAW,CAAC,OAAO,EAAE,MAAM,GAAG,YAAY;IAU1C,gCAAgC;IAC1B,eAAe,IAAI,OAAO,CAAC;QAAE,GAAG,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI,CAAC;IAMtE,4EAA4E;YAC9D,oBAAoB;CAyBnC"}
|
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
import { readFile, writeFile, mkdir } from 'node:fs/promises';
|
|
2
|
+
import { existsSync } from 'node:fs';
|
|
3
|
+
import { join } from 'node:path';
|
|
4
|
+
import { createHash, generateKeyPairSync, X509Certificate } from 'node:crypto';
|
|
5
|
+
import { homedir } from 'node:os';
|
|
6
|
+
import { CONFIG_DIR, TLS_DIR, TLS_CERT_FILE, TLS_KEY_FILE } from '@loopsy/protocol';
|
|
7
|
+
export class TlsManager {
|
|
8
|
+
dataDir;
|
|
9
|
+
tlsDir;
|
|
10
|
+
certPath;
|
|
11
|
+
keyPath;
|
|
12
|
+
constructor(dataDir) {
|
|
13
|
+
this.dataDir = dataDir ?? join(homedir(), CONFIG_DIR);
|
|
14
|
+
this.tlsDir = join(this.dataDir, TLS_DIR);
|
|
15
|
+
this.certPath = join(this.tlsDir, TLS_CERT_FILE);
|
|
16
|
+
this.keyPath = join(this.tlsDir, TLS_KEY_FILE);
|
|
17
|
+
}
|
|
18
|
+
/** Check if TLS cert/key already exist */
|
|
19
|
+
hasCerts() {
|
|
20
|
+
return existsSync(this.certPath) && existsSync(this.keyPath);
|
|
21
|
+
}
|
|
22
|
+
/** Generate a self-signed EC certificate if none exists */
|
|
23
|
+
async ensureCerts(hostname) {
|
|
24
|
+
if (this.hasCerts()) {
|
|
25
|
+
return this.loadCerts();
|
|
26
|
+
}
|
|
27
|
+
return this.generateCerts(hostname);
|
|
28
|
+
}
|
|
29
|
+
/** Generate a new self-signed certificate */
|
|
30
|
+
async generateCerts(hostname) {
|
|
31
|
+
await mkdir(this.tlsDir, { recursive: true });
|
|
32
|
+
// Generate EC keypair (P-256)
|
|
33
|
+
const { publicKey, privateKey } = generateKeyPairSync('ec', {
|
|
34
|
+
namedCurve: 'prime256v1',
|
|
35
|
+
publicKeyEncoding: { type: 'spki', format: 'pem' },
|
|
36
|
+
privateKeyEncoding: { type: 'pkcs8', format: 'pem' },
|
|
37
|
+
});
|
|
38
|
+
// Use Node's built-in X509 cert generation (available since Node 20)
|
|
39
|
+
// We'll use a child process with openssl as a fallback-free approach:
|
|
40
|
+
// Actually, Node doesn't have cert generation built-in. We'll generate
|
|
41
|
+
// a self-signed cert using the `selfsigned` approach via raw crypto.
|
|
42
|
+
//
|
|
43
|
+
// For simplicity, we'll shell out to openssl or use the node:crypto
|
|
44
|
+
// createSelfSignedCert if available. Since Node 22+ doesn't have this,
|
|
45
|
+
// we'll create a minimal self-signed cert with raw ASN.1.
|
|
46
|
+
const cert = await this.createSelfSignedCert(hostname, publicKey, privateKey);
|
|
47
|
+
await writeFile(this.keyPath, privateKey, { mode: 0o600 });
|
|
48
|
+
await writeFile(this.certPath, cert);
|
|
49
|
+
const fingerprint = this.computeFingerprint(cert);
|
|
50
|
+
return { cert, key: privateKey, fingerprint };
|
|
51
|
+
}
|
|
52
|
+
/** Load existing cert/key files */
|
|
53
|
+
async loadCerts() {
|
|
54
|
+
const cert = await readFile(this.certPath, 'utf-8');
|
|
55
|
+
const key = await readFile(this.keyPath, 'utf-8');
|
|
56
|
+
const fingerprint = this.computeFingerprint(cert);
|
|
57
|
+
return { cert, key, fingerprint };
|
|
58
|
+
}
|
|
59
|
+
/** Compute SHA-256 fingerprint of a PEM certificate */
|
|
60
|
+
computeFingerprint(certPem) {
|
|
61
|
+
// Extract DER from PEM
|
|
62
|
+
const b64 = certPem
|
|
63
|
+
.replace(/-----BEGIN CERTIFICATE-----/g, '')
|
|
64
|
+
.replace(/-----END CERTIFICATE-----/g, '')
|
|
65
|
+
.replace(/\s/g, '');
|
|
66
|
+
const der = Buffer.from(b64, 'base64');
|
|
67
|
+
return createHash('sha256').update(der).digest('hex');
|
|
68
|
+
}
|
|
69
|
+
/** Get cert info for display */
|
|
70
|
+
getCertInfo(certPem) {
|
|
71
|
+
const x509 = new X509Certificate(certPem);
|
|
72
|
+
return {
|
|
73
|
+
fingerprint: this.computeFingerprint(certPem),
|
|
74
|
+
hostname: x509.subject.split('CN=')[1]?.split('\n')[0] || 'unknown',
|
|
75
|
+
validFrom: x509.validFrom,
|
|
76
|
+
validTo: x509.validTo,
|
|
77
|
+
};
|
|
78
|
+
}
|
|
79
|
+
/** Get Fastify HTTPS options */
|
|
80
|
+
async getHttpsOptions() {
|
|
81
|
+
if (!this.hasCerts())
|
|
82
|
+
return null;
|
|
83
|
+
const { cert, key } = await this.loadCerts();
|
|
84
|
+
return { key, cert };
|
|
85
|
+
}
|
|
86
|
+
/** Create a self-signed certificate using openssl CLI (widely available) */
|
|
87
|
+
async createSelfSignedCert(hostname, _publicKey, privateKey) {
|
|
88
|
+
// Use Node's child_process to call openssl for cert generation
|
|
89
|
+
// This is the most portable approach that avoids native dependencies
|
|
90
|
+
const { execSync } = await import('node:child_process');
|
|
91
|
+
const { writeFileSync, readFileSync, unlinkSync } = await import('node:fs');
|
|
92
|
+
const { tmpdir } = await import('node:os');
|
|
93
|
+
const tmpKey = join(tmpdir(), `loopsy-key-${Date.now()}.pem`);
|
|
94
|
+
const tmpCert = join(tmpdir(), `loopsy-cert-${Date.now()}.pem`);
|
|
95
|
+
try {
|
|
96
|
+
writeFileSync(tmpKey, privateKey, { mode: 0o600 });
|
|
97
|
+
execSync(`openssl req -new -x509 -key "${tmpKey}" -out "${tmpCert}" ` +
|
|
98
|
+
`-days 3650 -subj "/CN=${hostname}/O=Loopsy" ` +
|
|
99
|
+
`-addext "subjectAltName=DNS:${hostname},DNS:localhost,IP:127.0.0.1"`, { stdio: 'pipe' });
|
|
100
|
+
return readFileSync(tmpCert, 'utf-8');
|
|
101
|
+
}
|
|
102
|
+
finally {
|
|
103
|
+
try {
|
|
104
|
+
unlinkSync(tmpKey);
|
|
105
|
+
}
|
|
106
|
+
catch { }
|
|
107
|
+
try {
|
|
108
|
+
unlinkSync(tmpCert);
|
|
109
|
+
}
|
|
110
|
+
catch { }
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
//# sourceMappingURL=tls-manager.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tls-manager.js","sourceRoot":"","sources":["../../src/services/tls-manager.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"which.d.ts","sourceRoot":"","sources":["../../src/utils/which.ts"],"names":[],"mappings":"AAGA,wBAAsB,KAAK,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAcnE"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import { access, constants } from 'node:fs/promises';
|
|
2
|
+
import { join } from 'node:path';
|
|
3
|
+
export async function which(command) {
|
|
4
|
+
const paths = (process.env.PATH || '').split(process.platform === 'win32' ? ';' : ':');
|
|
5
|
+
const extensions = process.platform === 'win32' ? ['.cmd', '.exe', '.bat', ''] : [''];
|
|
6
|
+
for (const dir of paths) {
|
|
7
|
+
for (const ext of extensions) {
|
|
8
|
+
const full = join(dir, command + ext);
|
|
9
|
+
try {
|
|
10
|
+
await access(full, constants.X_OK);
|
|
11
|
+
return full;
|
|
12
|
+
}
|
|
13
|
+
catch { }
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
return null;
|
|
17
|
+
}
|
|
18
|
+
//# sourceMappingURL=which.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"which.js","sourceRoot":"","sources":["../../src/utils/which.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,kBAAkB,CAAC;AACrD,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAEjC,MAAM,CAAC,KAAK,UAAU,KAAK,CAAC,OAAe;IACzC,MAAM,KAAK,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,IAAI,EAAE,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,KAAK,OAAO,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;IACvF,MAAM,UAAU,GAAG,OAAO,CAAC,QAAQ,KAAK,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;IAEtF,KAAK,MAAM,GAAG,IAAI,KAAK,EAAE,CAAC;QACxB,KAAK,MAAM,GAAG,IAAI,UAAU,EAAE,CAAC;YAC7B,MAAM,IAAI,GAAG,IAAI,CAAC,GAAG,EAAE,OAAO,GAAG,GAAG,CAAC,CAAC;YACtC,IAAI,CAAC;gBACH,MAAM,MAAM,CAAC,IAAI,EAAE,SAAS,CAAC,IAAI,CAAC,CAAC;gBACnC,OAAO,IAAI,CAAC;YACd,CAAC;YAAC,MAAM,CAAC,CAAA,CAAC;QACZ,CAAC;IACH,CAAC;IACD,OAAO,IAAI,CAAC;AACd,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAMA,MAAM,WAAW,eAAe;IAC9B,MAAM,EAAE,MAAM,CAAC;IACf,QAAQ,EAAE,MAAM,CAAC;IACjB,WAAW,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IACpC,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB;AAED,wBAAsB,mBAAmB,IAAI,OAAO,CAAC,eAAe,CAAC,CAcpE"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import { readFile } from 'node:fs/promises';
|
|
2
|
+
import { join } from 'node:path';
|
|
3
|
+
import { homedir } from 'node:os';
|
|
4
|
+
import { parse as parseYaml } from 'yaml';
|
|
5
|
+
import { CONFIG_DIR, CONFIG_FILE, DEFAULT_PORT } from '@loopsy/protocol';
|
|
6
|
+
export async function loadDashboardConfig() {
|
|
7
|
+
const configPath = join(homedir(), CONFIG_DIR, CONFIG_FILE);
|
|
8
|
+
try {
|
|
9
|
+
const raw = await readFile(configPath, 'utf-8');
|
|
10
|
+
const parsed = parseYaml(raw);
|
|
11
|
+
return {
|
|
12
|
+
apiKey: parsed?.auth?.apiKey ?? '',
|
|
13
|
+
mainPort: parsed?.server?.port ?? DEFAULT_PORT,
|
|
14
|
+
allowedKeys: parsed?.auth?.allowedKeys ?? {},
|
|
15
|
+
hostname: parsed?.server?.hostname,
|
|
16
|
+
};
|
|
17
|
+
}
|
|
18
|
+
catch {
|
|
19
|
+
return { apiKey: '', mainPort: DEFAULT_PORT, allowedKeys: {} };
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
//# sourceMappingURL=config.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.js","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,kBAAkB,CAAC;AAC5C,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,KAAK,IAAI,SAAS,EAAE,MAAM,MAAM,CAAC;AAC1C,OAAO,EAAE,UAAU,EAAE,WAAW,EAAE,YAAY,EAAE,MAAM,kBAAkB,CAAC;AASzE,MAAM,CAAC,KAAK,UAAU,mBAAmB;IACvC,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO,EAAE,EAAE,UAAU,EAAE,WAAW,CAAC,CAAC;IAC5D,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,MAAM,QAAQ,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;QAChD,MAAM,MAAM,GAAG,SAAS,CAAC,GAAG,CAAQ,CAAC;QACrC,OAAO;YACL,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,IAAI,EAAE;YAClC,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,IAAI,IAAI,YAAY;YAC9C,WAAW,EAAE,MAAM,EAAE,IAAI,EAAE,WAAW,IAAI,EAAE;YAC5C,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ;SACnC,CAAC;IACJ,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,MAAM,EAAE,EAAE,EAAE,QAAQ,EAAE,YAAY,EAAE,WAAW,EAAE,EAAE,EAAE,CAAC;IACjE,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1,120 @@
|
|
|
1
|
+
// ═══════════════════════════════════════════
|
|
2
|
+
// LOOPSY // DASHBOARD — App Router
|
|
3
|
+
// ═══════════════════════════════════════════
|
|
4
|
+
|
|
5
|
+
// --- API Helpers ---
|
|
6
|
+
|
|
7
|
+
export async function api(port, path, opts = {}) {
|
|
8
|
+
const headers = opts.body ? { 'Content-Type': 'application/json' } : {};
|
|
9
|
+
const res = await fetch(`/dashboard/api/proxy/${port}/api/v1${path}`, {
|
|
10
|
+
headers,
|
|
11
|
+
...opts,
|
|
12
|
+
});
|
|
13
|
+
if (!res.ok) {
|
|
14
|
+
const body = await res.json().catch(() => ({}));
|
|
15
|
+
throw new Error(body.error || `HTTP ${res.status}`);
|
|
16
|
+
}
|
|
17
|
+
return res.json();
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export async function dashboardApi(path, opts = {}) {
|
|
21
|
+
const headers = opts.body ? { 'Content-Type': 'application/json' } : {};
|
|
22
|
+
const res = await fetch(`/dashboard/api${path}`, {
|
|
23
|
+
headers,
|
|
24
|
+
...opts,
|
|
25
|
+
});
|
|
26
|
+
if (!res.ok) {
|
|
27
|
+
const body = await res.json().catch(() => ({}));
|
|
28
|
+
throw new Error(body.error || `HTTP ${res.status}`);
|
|
29
|
+
}
|
|
30
|
+
return res.json();
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
export function formatUptime(ms) {
|
|
34
|
+
if (!ms || ms < 0) return '—';
|
|
35
|
+
const s = Math.floor(ms / 1000);
|
|
36
|
+
const m = Math.floor(s / 60);
|
|
37
|
+
const h = Math.floor(m / 60);
|
|
38
|
+
const d = Math.floor(h / 24);
|
|
39
|
+
if (d > 0) return `${d}d ${h % 24}h`;
|
|
40
|
+
if (h > 0) return `${h}h ${m % 60}m`;
|
|
41
|
+
if (m > 0) return `${m}m ${s % 60}s`;
|
|
42
|
+
return `${s}s`;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
export function formatTime(ts) {
|
|
46
|
+
if (!ts) return '—';
|
|
47
|
+
return new Date(ts).toLocaleTimeString();
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
export function escapeHtml(str) {
|
|
51
|
+
const div = document.createElement('div');
|
|
52
|
+
div.textContent = str;
|
|
53
|
+
return div.innerHTML;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
// --- View Registry ---
|
|
57
|
+
|
|
58
|
+
const views = {};
|
|
59
|
+
let currentView = null;
|
|
60
|
+
let currentViewName = null;
|
|
61
|
+
|
|
62
|
+
export function registerView(name, view) {
|
|
63
|
+
views[name] = view;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
function navigate(viewName) {
|
|
67
|
+
if (currentView && currentView.unmount) currentView.unmount();
|
|
68
|
+
|
|
69
|
+
// Update nav
|
|
70
|
+
document.querySelectorAll('.nav-item, .mobile-nav-item').forEach(el => {
|
|
71
|
+
el.classList.toggle('active', el.dataset.view === viewName);
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
const main = document.getElementById('main');
|
|
75
|
+
main.innerHTML = '';
|
|
76
|
+
currentView = views[viewName];
|
|
77
|
+
currentViewName = viewName;
|
|
78
|
+
|
|
79
|
+
if (currentView && currentView.mount) {
|
|
80
|
+
currentView.mount(main);
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
history.replaceState(null, '', `#${viewName}`);
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// --- Clock ---
|
|
87
|
+
|
|
88
|
+
function updateClock() {
|
|
89
|
+
const el = document.getElementById('header-time');
|
|
90
|
+
if (el) el.textContent = new Date().toLocaleTimeString();
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
// --- Init ---
|
|
94
|
+
|
|
95
|
+
async function init() {
|
|
96
|
+
// Load view modules
|
|
97
|
+
await Promise.all([
|
|
98
|
+
import('/views/overview.js'),
|
|
99
|
+
import('/views/terminal.js'),
|
|
100
|
+
import('/views/context.js'),
|
|
101
|
+
import('/views/messages.js'),
|
|
102
|
+
import('/views/peers.js'),
|
|
103
|
+
import('/views/ai-tasks.js'),
|
|
104
|
+
]);
|
|
105
|
+
|
|
106
|
+
// Nav click handlers
|
|
107
|
+
document.querySelectorAll('.nav-item, .mobile-nav-item').forEach(el => {
|
|
108
|
+
el.addEventListener('click', () => navigate(el.dataset.view));
|
|
109
|
+
});
|
|
110
|
+
|
|
111
|
+
// Clock
|
|
112
|
+
updateClock();
|
|
113
|
+
setInterval(updateClock, 1000);
|
|
114
|
+
|
|
115
|
+
// Navigate to hash or default
|
|
116
|
+
const hash = location.hash.replace('#', '') || 'overview';
|
|
117
|
+
navigate(hash);
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
init();
|
|
Binary file
|
|
Binary file
|