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,49 @@
|
|
|
1
|
+
import { LoopsyError, LoopsyErrorCode } from '@loopsy/protocol';
|
|
2
|
+
export function registerContextRoutes(app, contextStore) {
|
|
3
|
+
app.put('/api/v1/context/:key', async (request, reply) => {
|
|
4
|
+
try {
|
|
5
|
+
const { key } = request.params;
|
|
6
|
+
const body = request.body;
|
|
7
|
+
if (!body?.value) {
|
|
8
|
+
throw new LoopsyError(LoopsyErrorCode.INVALID_REQUEST, 'Missing value');
|
|
9
|
+
}
|
|
10
|
+
const fromNodeId = request.headers['x-loopsy-node-id'] ?? 'local';
|
|
11
|
+
const entry = contextStore.set(key, body.value, fromNodeId, body.ttl);
|
|
12
|
+
await contextStore.save();
|
|
13
|
+
return entry;
|
|
14
|
+
}
|
|
15
|
+
catch (err) {
|
|
16
|
+
if (err instanceof LoopsyError) {
|
|
17
|
+
reply.code(400);
|
|
18
|
+
return err.toJSON();
|
|
19
|
+
}
|
|
20
|
+
throw err;
|
|
21
|
+
}
|
|
22
|
+
});
|
|
23
|
+
app.get('/api/v1/context/:key', async (request, reply) => {
|
|
24
|
+
const { key } = request.params;
|
|
25
|
+
const entry = contextStore.get(key);
|
|
26
|
+
if (!entry) {
|
|
27
|
+
const err = new LoopsyError(LoopsyErrorCode.CONTEXT_KEY_NOT_FOUND, `Key '${key}' not found`);
|
|
28
|
+
reply.code(404);
|
|
29
|
+
return err.toJSON();
|
|
30
|
+
}
|
|
31
|
+
return entry;
|
|
32
|
+
});
|
|
33
|
+
app.delete('/api/v1/context/:key', async (request, reply) => {
|
|
34
|
+
const { key } = request.params;
|
|
35
|
+
const deleted = contextStore.delete(key);
|
|
36
|
+
if (!deleted) {
|
|
37
|
+
const err = new LoopsyError(LoopsyErrorCode.CONTEXT_KEY_NOT_FOUND, `Key '${key}' not found`);
|
|
38
|
+
reply.code(404);
|
|
39
|
+
return err.toJSON();
|
|
40
|
+
}
|
|
41
|
+
await contextStore.save();
|
|
42
|
+
return { success: true, key };
|
|
43
|
+
});
|
|
44
|
+
app.get('/api/v1/context', async (request) => {
|
|
45
|
+
const { prefix } = request.query;
|
|
46
|
+
return { entries: contextStore.list(prefix) };
|
|
47
|
+
});
|
|
48
|
+
}
|
|
49
|
+
//# sourceMappingURL=context.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"context.js","sourceRoot":"","sources":["../../src/routes/context.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,WAAW,EAAE,eAAe,EAAE,MAAM,kBAAkB,CAAC;AAGhE,MAAM,UAAU,qBAAqB,CAAC,GAAoB,EAAE,YAA0B;IACpF,GAAG,CAAC,GAAG,CAA8B,sBAAsB,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE;QACpF,IAAI,CAAC;YACH,MAAM,EAAE,GAAG,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC;YAC/B,MAAM,IAAI,GAAG,OAAO,CAAC,IAAuC,CAAC;YAC7D,IAAI,CAAC,IAAI,EAAE,KAAK,EAAE,CAAC;gBACjB,MAAM,IAAI,WAAW,CAAC,eAAe,CAAC,eAAe,EAAE,eAAe,CAAC,CAAC;YAC1E,CAAC;YACD,MAAM,UAAU,GAAI,OAAO,CAAC,OAAO,CAAC,kBAAkB,CAAY,IAAI,OAAO,CAAC;YAC9E,MAAM,KAAK,GAAG,YAAY,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,KAAK,EAAE,UAAU,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC;YACtE,MAAM,YAAY,CAAC,IAAI,EAAE,CAAC;YAC1B,OAAO,KAAK,CAAC;QACf,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,IAAI,GAAG,YAAY,WAAW,EAAE,CAAC;gBAC/B,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBAChB,OAAO,GAAG,CAAC,MAAM,EAAE,CAAC;YACtB,CAAC;YACD,MAAM,GAAG,CAAC;QACZ,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAA8B,sBAAsB,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE;QACpF,MAAM,EAAE,GAAG,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC;QAC/B,MAAM,KAAK,GAAG,YAAY,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QACpC,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,MAAM,GAAG,GAAG,IAAI,WAAW,CAAC,eAAe,CAAC,qBAAqB,EAAE,QAAQ,GAAG,aAAa,CAAC,CAAC;YAC7F,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,GAAG,CAAC,MAAM,EAAE,CAAC;QACtB,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,MAAM,CAA8B,sBAAsB,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE;QACvF,MAAM,EAAE,GAAG,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC;QAC/B,MAAM,OAAO,GAAG,YAAY,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QACzC,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,MAAM,GAAG,GAAG,IAAI,WAAW,CAAC,eAAe,CAAC,qBAAqB,EAAE,QAAQ,GAAG,aAAa,CAAC,CAAC;YAC7F,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,GAAG,CAAC,MAAM,EAAE,CAAC;QACtB,CAAC;QACD,MAAM,YAAY,CAAC,IAAI,EAAE,CAAC;QAC1B,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,GAAG,EAAE,CAAC;IAChC,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAAC,iBAAiB,EAAE,KAAK,EAAE,OAAO,EAAE,EAAE;QAC3C,MAAM,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,KAA4B,CAAC;QACxD,OAAO,EAAE,OAAO,EAAE,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC;IAChD,CAAC,CAAC,CAAC;AACL,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"execute.d.ts","sourceRoot":"","sources":["../../src/routes/execute.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAE/C,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,4BAA4B,CAAC;AAE7D,wBAAgB,qBAAqB,CAAC,GAAG,EAAE,eAAe,EAAE,UAAU,EAAE,UAAU,QAgFjF"}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import { ExecuteParamsSchema, LoopsyError, LoopsyErrorCode } from '@loopsy/protocol';
|
|
2
|
+
export function registerExecuteRoutes(app, jobManager) {
|
|
3
|
+
app.post('/api/v1/execute', async (request, reply) => {
|
|
4
|
+
try {
|
|
5
|
+
const params = ExecuteParamsSchema.parse(request.body);
|
|
6
|
+
const fromNodeId = request.headers['x-loopsy-node-id'] ?? 'unknown';
|
|
7
|
+
const result = await jobManager.execute(params, fromNodeId);
|
|
8
|
+
return result;
|
|
9
|
+
}
|
|
10
|
+
catch (err) {
|
|
11
|
+
if (err instanceof LoopsyError) {
|
|
12
|
+
reply.code(err.code >= 3000 && err.code < 4000 ? 400 : 500);
|
|
13
|
+
return err.toJSON();
|
|
14
|
+
}
|
|
15
|
+
throw err;
|
|
16
|
+
}
|
|
17
|
+
});
|
|
18
|
+
app.post('/api/v1/execute/stream', async (request, reply) => {
|
|
19
|
+
try {
|
|
20
|
+
const params = ExecuteParamsSchema.parse(request.body);
|
|
21
|
+
const fromNodeId = request.headers['x-loopsy-node-id'] ?? 'unknown';
|
|
22
|
+
reply.raw.writeHead(200, {
|
|
23
|
+
'Content-Type': 'text/event-stream',
|
|
24
|
+
'Cache-Control': 'no-cache',
|
|
25
|
+
Connection: 'keep-alive',
|
|
26
|
+
});
|
|
27
|
+
const { spawn } = await import('node:child_process');
|
|
28
|
+
const proc = spawn(params.command, params.args ?? [], {
|
|
29
|
+
cwd: params.cwd,
|
|
30
|
+
env: params.env ? { ...process.env, ...params.env } : process.env,
|
|
31
|
+
shell: false,
|
|
32
|
+
timeout: params.timeout,
|
|
33
|
+
});
|
|
34
|
+
const jobId = `stream-${Date.now()}`;
|
|
35
|
+
const sendEvent = (type, data) => {
|
|
36
|
+
reply.raw.write(`data: ${JSON.stringify({ type, data, jobId, timestamp: Date.now() })}\n\n`);
|
|
37
|
+
};
|
|
38
|
+
proc.stdout?.on('data', (chunk) => sendEvent('stdout', chunk.toString()));
|
|
39
|
+
proc.stderr?.on('data', (chunk) => sendEvent('stderr', chunk.toString()));
|
|
40
|
+
proc.on('close', (exitCode) => {
|
|
41
|
+
sendEvent('exit', String(exitCode ?? 0));
|
|
42
|
+
reply.raw.end();
|
|
43
|
+
});
|
|
44
|
+
proc.on('error', (err) => {
|
|
45
|
+
sendEvent('error', err.message);
|
|
46
|
+
reply.raw.end();
|
|
47
|
+
});
|
|
48
|
+
request.raw.on('close', () => {
|
|
49
|
+
proc.kill('SIGTERM');
|
|
50
|
+
});
|
|
51
|
+
}
|
|
52
|
+
catch (err) {
|
|
53
|
+
if (err instanceof LoopsyError) {
|
|
54
|
+
reply.code(400);
|
|
55
|
+
return err.toJSON();
|
|
56
|
+
}
|
|
57
|
+
throw err;
|
|
58
|
+
}
|
|
59
|
+
});
|
|
60
|
+
app.delete('/api/v1/execute/:jobId', async (request, reply) => {
|
|
61
|
+
const { jobId } = request.params;
|
|
62
|
+
const cancelled = jobManager.cancel(jobId);
|
|
63
|
+
if (!cancelled) {
|
|
64
|
+
const err = new LoopsyError(LoopsyErrorCode.EXEC_JOB_NOT_FOUND, `Job ${jobId} not found`);
|
|
65
|
+
reply.code(404);
|
|
66
|
+
return err.toJSON();
|
|
67
|
+
}
|
|
68
|
+
return { success: true, jobId };
|
|
69
|
+
});
|
|
70
|
+
app.get('/api/v1/execute/jobs', async () => {
|
|
71
|
+
return { jobs: jobManager.getActiveJobs() };
|
|
72
|
+
});
|
|
73
|
+
}
|
|
74
|
+
//# sourceMappingURL=execute.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"execute.js","sourceRoot":"","sources":["../../src/routes/execute.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import type { FastifyInstance } from 'fastify';
|
|
2
|
+
import type { LoopsyNodeIdentity } from '@loopsy/protocol';
|
|
3
|
+
import type { PeerRegistry } from '@loopsy/discovery';
|
|
4
|
+
import type { JobManager } from '../services/job-manager.js';
|
|
5
|
+
import type { ContextStore } from '../services/context-store.js';
|
|
6
|
+
interface HealthRouteDeps {
|
|
7
|
+
identity: LoopsyNodeIdentity;
|
|
8
|
+
startTime: number;
|
|
9
|
+
registry: PeerRegistry;
|
|
10
|
+
jobManager: JobManager;
|
|
11
|
+
contextStore: ContextStore;
|
|
12
|
+
}
|
|
13
|
+
export declare function registerHealthRoutes(app: FastifyInstance, deps: HealthRouteDeps): void;
|
|
14
|
+
export {};
|
|
15
|
+
//# sourceMappingURL=health.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"health.d.ts","sourceRoot":"","sources":["../../src/routes/health.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAC/C,OAAO,KAAK,EAAE,kBAAkB,EAAkC,MAAM,kBAAkB,CAAC;AAE3F,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AACtD,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,4BAA4B,CAAC;AAC7D,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,8BAA8B,CAAC;AAEjE,UAAU,eAAe;IACvB,QAAQ,EAAE,kBAAkB,CAAC;IAC7B,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE,YAAY,CAAC;IACvB,UAAU,EAAE,UAAU,CAAC;IACvB,YAAY,EAAE,YAAY,CAAC;CAC5B;AAED,wBAAgB,oBAAoB,CAAC,GAAG,EAAE,eAAe,EAAE,IAAI,EAAE,eAAe,QAqC/E"}
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import { PROTOCOL_VERSION } from '@loopsy/protocol';
|
|
2
|
+
export function registerHealthRoutes(app, deps) {
|
|
3
|
+
app.get('/api/v1/health', async () => {
|
|
4
|
+
return {
|
|
5
|
+
status: 'ok',
|
|
6
|
+
nodeId: deps.identity.nodeId,
|
|
7
|
+
uptime: Date.now() - deps.startTime,
|
|
8
|
+
version: PROTOCOL_VERSION,
|
|
9
|
+
};
|
|
10
|
+
});
|
|
11
|
+
app.get('/api/v1/identity', async () => {
|
|
12
|
+
return deps.identity;
|
|
13
|
+
});
|
|
14
|
+
app.get('/api/v1/status', async () => {
|
|
15
|
+
const peers = deps.registry.getAll();
|
|
16
|
+
const onlinePeers = peers.filter((p) => p.status === 'online');
|
|
17
|
+
return {
|
|
18
|
+
nodeId: deps.identity.nodeId,
|
|
19
|
+
hostname: deps.identity.hostname,
|
|
20
|
+
platform: deps.identity.platform,
|
|
21
|
+
version: PROTOCOL_VERSION,
|
|
22
|
+
uptime: Date.now() - deps.startTime,
|
|
23
|
+
peers: {
|
|
24
|
+
total: peers.length,
|
|
25
|
+
online: onlinePeers.length,
|
|
26
|
+
offline: peers.length - onlinePeers.length,
|
|
27
|
+
},
|
|
28
|
+
jobs: {
|
|
29
|
+
active: deps.jobManager.activeCount,
|
|
30
|
+
total: 0,
|
|
31
|
+
},
|
|
32
|
+
context: {
|
|
33
|
+
entries: deps.contextStore.size,
|
|
34
|
+
},
|
|
35
|
+
};
|
|
36
|
+
});
|
|
37
|
+
}
|
|
38
|
+
//# sourceMappingURL=health.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"health.js","sourceRoot":"","sources":["../../src/routes/health.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,gBAAgB,EAAE,MAAM,kBAAkB,CAAC;AAapD,MAAM,UAAU,oBAAoB,CAAC,GAAoB,EAAE,IAAqB;IAC9E,GAAG,CAAC,GAAG,CAAC,gBAAgB,EAAE,KAAK,IAA6B,EAAE;QAC5D,OAAO;YACL,MAAM,EAAE,IAAI;YACZ,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM;YAC5B,MAAM,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,SAAS;YACnC,OAAO,EAAE,gBAAgB;SAC1B,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAAC,kBAAkB,EAAE,KAAK,IAAI,EAAE;QACrC,OAAO,IAAI,CAAC,QAAQ,CAAC;IACvB,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAAC,gBAAgB,EAAE,KAAK,IAA6B,EAAE;QAC5D,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC;QACrC,MAAM,WAAW,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC;QAC/D,OAAO;YACL,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM;YAC5B,QAAQ,EAAE,IAAI,CAAC,QAAQ,CAAC,QAAQ;YAChC,QAAQ,EAAE,IAAI,CAAC,QAAQ,CAAC,QAAQ;YAChC,OAAO,EAAE,gBAAgB;YACzB,MAAM,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,SAAS;YACnC,KAAK,EAAE;gBACL,KAAK,EAAE,KAAK,CAAC,MAAM;gBACnB,MAAM,EAAE,WAAW,CAAC,MAAM;gBAC1B,OAAO,EAAE,KAAK,CAAC,MAAM,GAAG,WAAW,CAAC,MAAM;aAC3C;YACD,IAAI,EAAE;gBACJ,MAAM,EAAE,IAAI,CAAC,UAAU,CAAC,WAAW;gBACnC,KAAK,EAAE,CAAC;aACT;YACD,OAAO,EAAE;gBACP,OAAO,EAAE,IAAI,CAAC,YAAY,CAAC,IAAI;aAChC;SACF,CAAC;IACJ,CAAC,CAAC,CAAC;AACL,CAAC"}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import type { FastifyInstance } from 'fastify';
|
|
2
|
+
import type { TlsManager } from '../services/tls-manager.js';
|
|
3
|
+
interface PairContext {
|
|
4
|
+
hostname: string;
|
|
5
|
+
apiKey: string;
|
|
6
|
+
tlsManager: TlsManager;
|
|
7
|
+
dataDir?: string;
|
|
8
|
+
}
|
|
9
|
+
export declare function registerPairRoutes(app: FastifyInstance, ctx: PairContext): void;
|
|
10
|
+
export {};
|
|
11
|
+
//# sourceMappingURL=pair.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pair.d.ts","sourceRoot":"","sources":["../../src/routes/pair.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAQ/C,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,4BAA4B,CAAC;AAE7D,UAAU,WAAW;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,EAAE,UAAU,CAAC;IACvB,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED,wBAAgB,kBAAkB,CAAC,GAAG,EAAE,eAAe,EAAE,GAAG,EAAE,WAAW,QA+IxE"}
|
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
import { randomBytes, createECDH, createHash } from 'node:crypto';
|
|
2
|
+
import { readFile, writeFile } from 'node:fs/promises';
|
|
3
|
+
import { join } from 'node:path';
|
|
4
|
+
import { homedir } from 'node:os';
|
|
5
|
+
import { parse as parseYaml, stringify as toYaml } from 'yaml';
|
|
6
|
+
import { PAIRING_CODE_LENGTH, PAIRING_TIMEOUT, CONFIG_DIR, CONFIG_FILE } from '@loopsy/protocol';
|
|
7
|
+
export function registerPairRoutes(app, ctx) {
|
|
8
|
+
// Active pairing session (only one at a time)
|
|
9
|
+
let session = null;
|
|
10
|
+
let sessionEcdh = null;
|
|
11
|
+
let sessionTimeout = null;
|
|
12
|
+
// Store peer info during key exchange until confirmed
|
|
13
|
+
let pendingPeer = null;
|
|
14
|
+
// POST /api/v1/pair/start — Machine A starts a pairing session
|
|
15
|
+
app.post('/api/v1/pair/start', async (_request, reply) => {
|
|
16
|
+
if (session && session.state === 'waiting' && Date.now() < session.expiresAt) {
|
|
17
|
+
return reply.status(409).send({ error: 'Pairing session already active' });
|
|
18
|
+
}
|
|
19
|
+
// Generate ECDH keypair
|
|
20
|
+
const ecdh = createECDH('prime256v1');
|
|
21
|
+
ecdh.generateKeys();
|
|
22
|
+
// Generate invite code (6 digits)
|
|
23
|
+
const code = String(randomBytes(4).readUInt32BE() % 10 ** PAIRING_CODE_LENGTH).padStart(PAIRING_CODE_LENGTH, '0');
|
|
24
|
+
session = {
|
|
25
|
+
inviteCode: code,
|
|
26
|
+
publicKey: ecdh.getPublicKey('base64'),
|
|
27
|
+
expiresAt: Date.now() + PAIRING_TIMEOUT,
|
|
28
|
+
state: 'waiting',
|
|
29
|
+
};
|
|
30
|
+
sessionEcdh = ecdh;
|
|
31
|
+
pendingPeer = null;
|
|
32
|
+
// Auto-expire
|
|
33
|
+
if (sessionTimeout)
|
|
34
|
+
clearTimeout(sessionTimeout);
|
|
35
|
+
sessionTimeout = setTimeout(() => {
|
|
36
|
+
if (session)
|
|
37
|
+
session.state = 'expired';
|
|
38
|
+
session = null;
|
|
39
|
+
sessionEcdh = null;
|
|
40
|
+
pendingPeer = null;
|
|
41
|
+
}, PAIRING_TIMEOUT);
|
|
42
|
+
return { inviteCode: code, expiresAt: session.expiresAt };
|
|
43
|
+
});
|
|
44
|
+
// POST /api/v1/pair/initiate — Machine B sends its public key + invite code
|
|
45
|
+
app.post('/api/v1/pair/initiate', async (request, reply) => {
|
|
46
|
+
const body = request.body;
|
|
47
|
+
if (!session || session.state !== 'waiting') {
|
|
48
|
+
return reply.status(404).send({ error: 'No active pairing session' });
|
|
49
|
+
}
|
|
50
|
+
if (Date.now() > session.expiresAt) {
|
|
51
|
+
session.state = 'expired';
|
|
52
|
+
return reply.status(410).send({ error: 'Pairing session expired' });
|
|
53
|
+
}
|
|
54
|
+
if (body.inviteCode !== session.inviteCode) {
|
|
55
|
+
return reply.status(403).send({ error: 'Invalid invite code' });
|
|
56
|
+
}
|
|
57
|
+
if (!sessionEcdh) {
|
|
58
|
+
return reply.status(500).send({ error: 'Internal error: missing ECDH state' });
|
|
59
|
+
}
|
|
60
|
+
// Compute shared secret via ECDH
|
|
61
|
+
const peerPubKey = Buffer.from(body.publicKey, 'base64');
|
|
62
|
+
const sharedSecret = sessionEcdh.computeSecret(peerPubKey);
|
|
63
|
+
// Derive SAS (Short Authentication String) — 6 digits from shared secret hash
|
|
64
|
+
const sasHash = createHash('sha256').update(sharedSecret).update('loopsy-sas').digest();
|
|
65
|
+
const sas = String(sasHash.readUInt32BE() % 10 ** 6).padStart(6, '0');
|
|
66
|
+
session.peerPublicKey = body.publicKey;
|
|
67
|
+
session.sas = sas;
|
|
68
|
+
session.state = 'key_exchanged';
|
|
69
|
+
pendingPeer = {
|
|
70
|
+
hostname: body.hostname,
|
|
71
|
+
apiKey: body.apiKey,
|
|
72
|
+
certFingerprint: body.certFingerprint,
|
|
73
|
+
};
|
|
74
|
+
// Get our TLS fingerprint if available
|
|
75
|
+
let ourFingerprint;
|
|
76
|
+
if (ctx.tlsManager.hasCerts()) {
|
|
77
|
+
const certs = await ctx.tlsManager.loadCerts();
|
|
78
|
+
ourFingerprint = certs.fingerprint;
|
|
79
|
+
}
|
|
80
|
+
const response = {
|
|
81
|
+
publicKey: session.publicKey,
|
|
82
|
+
hostname: ctx.hostname,
|
|
83
|
+
apiKey: ctx.apiKey,
|
|
84
|
+
certFingerprint: ourFingerprint,
|
|
85
|
+
sas,
|
|
86
|
+
};
|
|
87
|
+
return response;
|
|
88
|
+
});
|
|
89
|
+
// POST /api/v1/pair/confirm — Both sides confirm the SAS matches
|
|
90
|
+
app.post('/api/v1/pair/confirm', async (request, reply) => {
|
|
91
|
+
const body = request.body;
|
|
92
|
+
if (!session || session.state !== 'key_exchanged') {
|
|
93
|
+
return reply.status(404).send({ error: 'No pairing session awaiting confirmation' });
|
|
94
|
+
}
|
|
95
|
+
if (!body.confirmed) {
|
|
96
|
+
session.state = 'expired';
|
|
97
|
+
session = null;
|
|
98
|
+
sessionEcdh = null;
|
|
99
|
+
pendingPeer = null;
|
|
100
|
+
return { success: false, message: 'Pairing cancelled' };
|
|
101
|
+
}
|
|
102
|
+
if (!pendingPeer) {
|
|
103
|
+
return reply.status(500).send({ error: 'No pending peer info' });
|
|
104
|
+
}
|
|
105
|
+
// Write peer to config
|
|
106
|
+
await addPeerToConfig(pendingPeer.hostname, pendingPeer.apiKey, pendingPeer.certFingerprint, ctx.dataDir);
|
|
107
|
+
session.state = 'completed';
|
|
108
|
+
const completedPeer = pendingPeer.hostname;
|
|
109
|
+
// Cleanup
|
|
110
|
+
session = null;
|
|
111
|
+
sessionEcdh = null;
|
|
112
|
+
pendingPeer = null;
|
|
113
|
+
if (sessionTimeout)
|
|
114
|
+
clearTimeout(sessionTimeout);
|
|
115
|
+
return { success: true, message: `Paired with ${completedPeer}` };
|
|
116
|
+
});
|
|
117
|
+
// GET /api/v1/pair/status — Check current pairing session state
|
|
118
|
+
app.get('/api/v1/pair/status', async () => {
|
|
119
|
+
if (!session)
|
|
120
|
+
return { active: false };
|
|
121
|
+
return {
|
|
122
|
+
active: true,
|
|
123
|
+
state: session.state,
|
|
124
|
+
sas: session.state === 'key_exchanged' ? session.sas : undefined,
|
|
125
|
+
expiresAt: session.expiresAt,
|
|
126
|
+
};
|
|
127
|
+
});
|
|
128
|
+
}
|
|
129
|
+
async function addPeerToConfig(peerHostname, peerApiKey, certFingerprint, dataDir) {
|
|
130
|
+
const configPath = join(dataDir ?? join(homedir(), CONFIG_DIR), CONFIG_FILE);
|
|
131
|
+
const raw = await readFile(configPath, 'utf-8');
|
|
132
|
+
const config = parseYaml(raw);
|
|
133
|
+
// Add to allowedKeys
|
|
134
|
+
if (!config.auth)
|
|
135
|
+
config.auth = {};
|
|
136
|
+
if (!config.auth.allowedKeys)
|
|
137
|
+
config.auth.allowedKeys = {};
|
|
138
|
+
config.auth.allowedKeys[peerHostname] = peerApiKey;
|
|
139
|
+
// Add cert fingerprint if provided
|
|
140
|
+
if (certFingerprint) {
|
|
141
|
+
if (!config.tls)
|
|
142
|
+
config.tls = { enabled: false };
|
|
143
|
+
if (!config.tls.pinnedCerts)
|
|
144
|
+
config.tls.pinnedCerts = {};
|
|
145
|
+
config.tls.pinnedCerts[peerHostname] = certFingerprint;
|
|
146
|
+
}
|
|
147
|
+
await writeFile(configPath, toYaml(config));
|
|
148
|
+
}
|
|
149
|
+
//# sourceMappingURL=pair.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pair.js","sourceRoot":"","sources":["../../src/routes/pair.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import type { FastifyInstance } from 'fastify';
|
|
2
|
+
import type { LoopsyNodeIdentity } from '@loopsy/protocol';
|
|
3
|
+
import type { PeerRegistry } from '@loopsy/discovery';
|
|
4
|
+
export declare function registerPeerRoutes(app: FastifyInstance, identity: LoopsyNodeIdentity, registry: PeerRegistry): void;
|
|
5
|
+
//# sourceMappingURL=peers.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"peers.d.ts","sourceRoot":"","sources":["../../src/routes/peers.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAE/C,OAAO,KAAK,EAAE,kBAAkB,EAAY,MAAM,kBAAkB,CAAC;AACrE,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AAEtD,wBAAgB,kBAAkB,CAChC,GAAG,EAAE,eAAe,EACpB,QAAQ,EAAE,kBAAkB,EAC5B,QAAQ,EAAE,YAAY,QAyEvB"}
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
import { HandshakeSchema, LoopsyError, LoopsyErrorCode } from '@loopsy/protocol';
|
|
2
|
+
export function registerPeerRoutes(app, identity, registry) {
|
|
3
|
+
// Handshake: receive a peer's identity
|
|
4
|
+
app.post('/api/v1/peers/handshake', async (request) => {
|
|
5
|
+
const data = HandshakeSchema.parse(request.body);
|
|
6
|
+
const address = request.ip;
|
|
7
|
+
const peer = {
|
|
8
|
+
nodeId: data.nodeId,
|
|
9
|
+
hostname: data.hostname,
|
|
10
|
+
address,
|
|
11
|
+
port: data.port,
|
|
12
|
+
platform: data.platform,
|
|
13
|
+
version: data.version,
|
|
14
|
+
capabilities: data.capabilities,
|
|
15
|
+
status: 'online',
|
|
16
|
+
lastSeen: Date.now(),
|
|
17
|
+
failureCount: 0,
|
|
18
|
+
trusted: false,
|
|
19
|
+
manuallyAdded: false,
|
|
20
|
+
};
|
|
21
|
+
registry.upsert(peer);
|
|
22
|
+
await registry.save();
|
|
23
|
+
return {
|
|
24
|
+
nodeId: identity.nodeId,
|
|
25
|
+
hostname: identity.hostname,
|
|
26
|
+
platform: identity.platform,
|
|
27
|
+
version: identity.version,
|
|
28
|
+
port: identity.port,
|
|
29
|
+
capabilities: identity.capabilities,
|
|
30
|
+
};
|
|
31
|
+
});
|
|
32
|
+
// List all known peers
|
|
33
|
+
app.get('/api/v1/peers', async () => {
|
|
34
|
+
return { peers: registry.getAll() };
|
|
35
|
+
});
|
|
36
|
+
// Add a manual peer
|
|
37
|
+
app.post('/api/v1/peers', async (request) => {
|
|
38
|
+
const body = request.body;
|
|
39
|
+
const peer = {
|
|
40
|
+
nodeId: body.nodeId ?? `manual-${body.address}:${body.port}`,
|
|
41
|
+
hostname: body.hostname || body.address,
|
|
42
|
+
address: body.address,
|
|
43
|
+
port: body.port,
|
|
44
|
+
platform: 'unknown',
|
|
45
|
+
version: 'unknown',
|
|
46
|
+
capabilities: [],
|
|
47
|
+
status: 'unknown',
|
|
48
|
+
lastSeen: 0,
|
|
49
|
+
failureCount: 0,
|
|
50
|
+
trusted: false,
|
|
51
|
+
manuallyAdded: true,
|
|
52
|
+
};
|
|
53
|
+
registry.upsert(peer);
|
|
54
|
+
await registry.save();
|
|
55
|
+
return peer;
|
|
56
|
+
});
|
|
57
|
+
// Remove a peer
|
|
58
|
+
app.delete('/api/v1/peers/:nodeId', async (request, reply) => {
|
|
59
|
+
const { nodeId } = request.params;
|
|
60
|
+
const removed = registry.remove(nodeId);
|
|
61
|
+
if (!removed) {
|
|
62
|
+
reply.code(404);
|
|
63
|
+
return new LoopsyError(LoopsyErrorCode.PEER_NOT_FOUND, `Peer ${nodeId} not found`).toJSON();
|
|
64
|
+
}
|
|
65
|
+
await registry.save();
|
|
66
|
+
return { success: true, nodeId };
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
//# sourceMappingURL=peers.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"peers.js","sourceRoot":"","sources":["../../src/routes/peers.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"transfer.d.ts","sourceRoot":"","sources":["../../src/routes/transfer.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAQ/C,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,kBAAkB,CAAC;AAErD,wBAAgB,sBAAsB,CAAC,GAAG,EAAE,eAAe,EAAE,MAAM,EAAE,YAAY,QA2IhF"}
|
|
@@ -0,0 +1,135 @@
|
|
|
1
|
+
import { createReadStream, createWriteStream } from 'node:fs';
|
|
2
|
+
import { stat, readdir, mkdir } from 'node:fs/promises';
|
|
3
|
+
import { join, dirname, resolve, normalize } from 'node:path';
|
|
4
|
+
import { createHash } from 'node:crypto';
|
|
5
|
+
import { LoopsyError, LoopsyErrorCode } from '@loopsy/protocol';
|
|
6
|
+
export function registerTransferRoutes(app, config) {
|
|
7
|
+
const isPathAllowed = (filePath) => {
|
|
8
|
+
const normalized = normalize(resolve(filePath));
|
|
9
|
+
for (const denied of config.transfer.deniedPaths) {
|
|
10
|
+
if (normalized.startsWith(normalize(resolve(denied))))
|
|
11
|
+
return false;
|
|
12
|
+
}
|
|
13
|
+
if (config.transfer.allowedPaths.length === 0)
|
|
14
|
+
return true;
|
|
15
|
+
for (const allowed of config.transfer.allowedPaths) {
|
|
16
|
+
if (normalized.startsWith(normalize(resolve(allowed))))
|
|
17
|
+
return true;
|
|
18
|
+
}
|
|
19
|
+
return false;
|
|
20
|
+
};
|
|
21
|
+
// Push: receive a file from a peer
|
|
22
|
+
app.post('/api/v1/transfer/push', async (request, reply) => {
|
|
23
|
+
try {
|
|
24
|
+
const data = await request.file();
|
|
25
|
+
if (!data) {
|
|
26
|
+
throw new LoopsyError(LoopsyErrorCode.TRANSFER_FAILED, 'No file in request');
|
|
27
|
+
}
|
|
28
|
+
const destPath = data.fields?.destPath?.value;
|
|
29
|
+
if (!destPath) {
|
|
30
|
+
throw new LoopsyError(LoopsyErrorCode.INVALID_REQUEST, 'Missing destPath field');
|
|
31
|
+
}
|
|
32
|
+
if (!isPathAllowed(destPath)) {
|
|
33
|
+
throw new LoopsyError(LoopsyErrorCode.TRANSFER_PATH_DENIED, `Path '${destPath}' is not allowed`);
|
|
34
|
+
}
|
|
35
|
+
await mkdir(dirname(destPath), { recursive: true });
|
|
36
|
+
const hash = createHash('sha256');
|
|
37
|
+
let size = 0;
|
|
38
|
+
const start = Date.now();
|
|
39
|
+
const writeStream = createWriteStream(destPath);
|
|
40
|
+
for await (const chunk of data.file) {
|
|
41
|
+
hash.update(chunk);
|
|
42
|
+
size += chunk.length;
|
|
43
|
+
if (size > config.transfer.maxFileSize) {
|
|
44
|
+
writeStream.destroy();
|
|
45
|
+
throw new LoopsyError(LoopsyErrorCode.TRANSFER_TOO_LARGE, `File exceeds max size of ${config.transfer.maxFileSize} bytes`);
|
|
46
|
+
}
|
|
47
|
+
writeStream.write(chunk);
|
|
48
|
+
}
|
|
49
|
+
writeStream.end();
|
|
50
|
+
const result = {
|
|
51
|
+
path: destPath,
|
|
52
|
+
size,
|
|
53
|
+
checksum: hash.digest('hex'),
|
|
54
|
+
duration: Date.now() - start,
|
|
55
|
+
};
|
|
56
|
+
return result;
|
|
57
|
+
}
|
|
58
|
+
catch (err) {
|
|
59
|
+
if (err instanceof LoopsyError) {
|
|
60
|
+
reply.code(400);
|
|
61
|
+
return err.toJSON();
|
|
62
|
+
}
|
|
63
|
+
throw err;
|
|
64
|
+
}
|
|
65
|
+
});
|
|
66
|
+
// Pull: send a file to a peer
|
|
67
|
+
app.post('/api/v1/transfer/pull', async (request, reply) => {
|
|
68
|
+
try {
|
|
69
|
+
const body = request.body;
|
|
70
|
+
if (!body?.sourcePath) {
|
|
71
|
+
throw new LoopsyError(LoopsyErrorCode.INVALID_REQUEST, 'Missing sourcePath');
|
|
72
|
+
}
|
|
73
|
+
if (!isPathAllowed(body.sourcePath)) {
|
|
74
|
+
throw new LoopsyError(LoopsyErrorCode.TRANSFER_PATH_DENIED, `Path '${body.sourcePath}' is not allowed`);
|
|
75
|
+
}
|
|
76
|
+
const fileStat = await stat(body.sourcePath).catch(() => null);
|
|
77
|
+
if (!fileStat || !fileStat.isFile()) {
|
|
78
|
+
throw new LoopsyError(LoopsyErrorCode.TRANSFER_FILE_NOT_FOUND, `File not found: ${body.sourcePath}`);
|
|
79
|
+
}
|
|
80
|
+
if (fileStat.size > config.transfer.maxFileSize) {
|
|
81
|
+
throw new LoopsyError(LoopsyErrorCode.TRANSFER_TOO_LARGE, `File exceeds max size`);
|
|
82
|
+
}
|
|
83
|
+
reply.header('Content-Type', 'application/octet-stream');
|
|
84
|
+
reply.header('Content-Disposition', `attachment; filename="${body.sourcePath.split('/').pop()}"`);
|
|
85
|
+
reply.header('X-Loopsy-File-Size', String(fileStat.size));
|
|
86
|
+
return reply.send(createReadStream(body.sourcePath));
|
|
87
|
+
}
|
|
88
|
+
catch (err) {
|
|
89
|
+
if (err instanceof LoopsyError) {
|
|
90
|
+
reply.code(400);
|
|
91
|
+
return err.toJSON();
|
|
92
|
+
}
|
|
93
|
+
throw err;
|
|
94
|
+
}
|
|
95
|
+
});
|
|
96
|
+
// List directory contents
|
|
97
|
+
app.post('/api/v1/transfer/list', async (request, reply) => {
|
|
98
|
+
try {
|
|
99
|
+
const body = request.body;
|
|
100
|
+
if (!body?.path) {
|
|
101
|
+
throw new LoopsyError(LoopsyErrorCode.INVALID_REQUEST, 'Missing path');
|
|
102
|
+
}
|
|
103
|
+
if (!isPathAllowed(body.path)) {
|
|
104
|
+
throw new LoopsyError(LoopsyErrorCode.TRANSFER_PATH_DENIED, `Path '${body.path}' is not allowed`);
|
|
105
|
+
}
|
|
106
|
+
const entries = await readdir(body.path, { withFileTypes: true });
|
|
107
|
+
const files = [];
|
|
108
|
+
for (const entry of entries) {
|
|
109
|
+
const fullPath = join(body.path, entry.name);
|
|
110
|
+
try {
|
|
111
|
+
const s = await stat(fullPath);
|
|
112
|
+
files.push({
|
|
113
|
+
name: entry.name,
|
|
114
|
+
path: fullPath,
|
|
115
|
+
type: entry.isDirectory() ? 'directory' : entry.isSymbolicLink() ? 'symlink' : 'file',
|
|
116
|
+
size: s.size,
|
|
117
|
+
modified: s.mtimeMs,
|
|
118
|
+
});
|
|
119
|
+
}
|
|
120
|
+
catch {
|
|
121
|
+
// Skip files we can't stat
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
return { files };
|
|
125
|
+
}
|
|
126
|
+
catch (err) {
|
|
127
|
+
if (err instanceof LoopsyError) {
|
|
128
|
+
reply.code(400);
|
|
129
|
+
return err.toJSON();
|
|
130
|
+
}
|
|
131
|
+
throw err;
|
|
132
|
+
}
|
|
133
|
+
});
|
|
134
|
+
}
|
|
135
|
+
//# sourceMappingURL=transfer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"transfer.js","sourceRoot":"","sources":["../../src/routes/transfer.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import type { LoopsyConfig } from '@loopsy/protocol';
|
|
2
|
+
export interface DaemonServer {
|
|
3
|
+
start(): Promise<void>;
|
|
4
|
+
stop(): Promise<void>;
|
|
5
|
+
getAddress(): string;
|
|
6
|
+
}
|
|
7
|
+
export declare function createDaemon(config: LoopsyConfig): Promise<DaemonServer>;
|
|
8
|
+
//# sourceMappingURL=server.d.ts.map
|