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.
Files changed (262) hide show
  1. package/LICENSE +190 -0
  2. package/README.md +425 -0
  3. package/dist/cli/commands/connect.d.ts +2 -0
  4. package/dist/cli/commands/connect.d.ts.map +1 -0
  5. package/dist/cli/commands/connect.js +120 -0
  6. package/dist/cli/commands/connect.js.map +1 -0
  7. package/dist/cli/commands/context.d.ts +2 -0
  8. package/dist/cli/commands/context.d.ts.map +1 -0
  9. package/dist/cli/commands/context.js +39 -0
  10. package/dist/cli/commands/context.js.map +1 -0
  11. package/dist/cli/commands/daemon.d.ts +4 -0
  12. package/dist/cli/commands/daemon.d.ts.map +1 -0
  13. package/dist/cli/commands/daemon.js +55 -0
  14. package/dist/cli/commands/daemon.js.map +1 -0
  15. package/dist/cli/commands/dashboard.d.ts +2 -0
  16. package/dist/cli/commands/dashboard.d.ts.map +1 -0
  17. package/dist/cli/commands/dashboard.js +24 -0
  18. package/dist/cli/commands/dashboard.js.map +1 -0
  19. package/dist/cli/commands/doctor.d.ts +2 -0
  20. package/dist/cli/commands/doctor.d.ts.map +1 -0
  21. package/dist/cli/commands/doctor.js +130 -0
  22. package/dist/cli/commands/doctor.js.map +1 -0
  23. package/dist/cli/commands/exec.d.ts +2 -0
  24. package/dist/cli/commands/exec.d.ts.map +1 -0
  25. package/dist/cli/commands/exec.js +34 -0
  26. package/dist/cli/commands/exec.js.map +1 -0
  27. package/dist/cli/commands/init.d.ts +2 -0
  28. package/dist/cli/commands/init.d.ts.map +1 -0
  29. package/dist/cli/commands/init.js +71 -0
  30. package/dist/cli/commands/init.js.map +1 -0
  31. package/dist/cli/commands/key.d.ts +2 -0
  32. package/dist/cli/commands/key.d.ts.map +1 -0
  33. package/dist/cli/commands/key.js +39 -0
  34. package/dist/cli/commands/key.js.map +1 -0
  35. package/dist/cli/commands/logs.d.ts +2 -0
  36. package/dist/cli/commands/logs.d.ts.map +1 -0
  37. package/dist/cli/commands/logs.js +26 -0
  38. package/dist/cli/commands/logs.js.map +1 -0
  39. package/dist/cli/commands/mcp.d.ts +4 -0
  40. package/dist/cli/commands/mcp.d.ts.map +1 -0
  41. package/dist/cli/commands/mcp.js +70 -0
  42. package/dist/cli/commands/mcp.js.map +1 -0
  43. package/dist/cli/commands/pair.d.ts +6 -0
  44. package/dist/cli/commands/pair.d.ts.map +1 -0
  45. package/dist/cli/commands/pair.js +208 -0
  46. package/dist/cli/commands/pair.js.map +1 -0
  47. package/dist/cli/commands/peers.d.ts +2 -0
  48. package/dist/cli/commands/peers.d.ts.map +1 -0
  49. package/dist/cli/commands/peers.js +29 -0
  50. package/dist/cli/commands/peers.js.map +1 -0
  51. package/dist/cli/commands/service/linux.d.ts +7 -0
  52. package/dist/cli/commands/service/linux.d.ts.map +1 -0
  53. package/dist/cli/commands/service/linux.js +86 -0
  54. package/dist/cli/commands/service/linux.js.map +1 -0
  55. package/dist/cli/commands/service/macos.d.ts +7 -0
  56. package/dist/cli/commands/service/macos.d.ts.map +1 -0
  57. package/dist/cli/commands/service/macos.js +83 -0
  58. package/dist/cli/commands/service/macos.js.map +1 -0
  59. package/dist/cli/commands/service/windows.d.ts +7 -0
  60. package/dist/cli/commands/service/windows.d.ts.map +1 -0
  61. package/dist/cli/commands/service/windows.js +52 -0
  62. package/dist/cli/commands/service/windows.js.map +1 -0
  63. package/dist/cli/commands/service.d.ts +4 -0
  64. package/dist/cli/commands/service.d.ts.map +1 -0
  65. package/dist/cli/commands/service.js +68 -0
  66. package/dist/cli/commands/service.js.map +1 -0
  67. package/dist/cli/commands/session.d.ts +8 -0
  68. package/dist/cli/commands/session.d.ts.map +1 -0
  69. package/dist/cli/commands/session.js +270 -0
  70. package/dist/cli/commands/session.js.map +1 -0
  71. package/dist/cli/commands/transfer.d.ts +3 -0
  72. package/dist/cli/commands/transfer.d.ts.map +1 -0
  73. package/dist/cli/commands/transfer.js +57 -0
  74. package/dist/cli/commands/transfer.js.map +1 -0
  75. package/dist/cli/index.d.ts +3 -0
  76. package/dist/cli/index.d.ts.map +1 -0
  77. package/dist/cli/index.js +89 -0
  78. package/dist/cli/index.js.map +1 -0
  79. package/dist/cli/package-root.d.ts +27 -0
  80. package/dist/cli/package-root.d.ts.map +1 -0
  81. package/dist/cli/package-root.js +54 -0
  82. package/dist/cli/package-root.js.map +1 -0
  83. package/dist/cli/utils.d.ts +11 -0
  84. package/dist/cli/utils.d.ts.map +1 -0
  85. package/dist/cli/utils.js +48 -0
  86. package/dist/cli/utils.js.map +1 -0
  87. package/dist/daemon/config.d.ts +4 -0
  88. package/dist/daemon/config.d.ts.map +1 -0
  89. package/dist/daemon/config.js +58 -0
  90. package/dist/daemon/config.js.map +1 -0
  91. package/dist/daemon/hooks/permission-hook.mjs +108 -0
  92. package/dist/daemon/index.d.ts +3 -0
  93. package/dist/daemon/index.d.ts.map +1 -0
  94. package/dist/daemon/index.js +3 -0
  95. package/dist/daemon/index.js.map +1 -0
  96. package/dist/daemon/main.d.ts +3 -0
  97. package/dist/daemon/main.d.ts.map +1 -0
  98. package/dist/daemon/main.js +28 -0
  99. package/dist/daemon/main.js.map +1 -0
  100. package/dist/daemon/middleware/auth.d.ts +3 -0
  101. package/dist/daemon/middleware/auth.d.ts.map +1 -0
  102. package/dist/daemon/middleware/auth.js +22 -0
  103. package/dist/daemon/middleware/auth.js.map +1 -0
  104. package/dist/daemon/routes/ai-tasks.d.ts +4 -0
  105. package/dist/daemon/routes/ai-tasks.d.ts.map +1 -0
  106. package/dist/daemon/routes/ai-tasks.js +146 -0
  107. package/dist/daemon/routes/ai-tasks.js.map +1 -0
  108. package/dist/daemon/routes/context.d.ts +4 -0
  109. package/dist/daemon/routes/context.d.ts.map +1 -0
  110. package/dist/daemon/routes/context.js +49 -0
  111. package/dist/daemon/routes/context.js.map +1 -0
  112. package/dist/daemon/routes/execute.d.ts +4 -0
  113. package/dist/daemon/routes/execute.d.ts.map +1 -0
  114. package/dist/daemon/routes/execute.js +74 -0
  115. package/dist/daemon/routes/execute.js.map +1 -0
  116. package/dist/daemon/routes/health.d.ts +15 -0
  117. package/dist/daemon/routes/health.d.ts.map +1 -0
  118. package/dist/daemon/routes/health.js +38 -0
  119. package/dist/daemon/routes/health.js.map +1 -0
  120. package/dist/daemon/routes/pair.d.ts +11 -0
  121. package/dist/daemon/routes/pair.d.ts.map +1 -0
  122. package/dist/daemon/routes/pair.js +149 -0
  123. package/dist/daemon/routes/pair.js.map +1 -0
  124. package/dist/daemon/routes/peers.d.ts +5 -0
  125. package/dist/daemon/routes/peers.d.ts.map +1 -0
  126. package/dist/daemon/routes/peers.js +69 -0
  127. package/dist/daemon/routes/peers.js.map +1 -0
  128. package/dist/daemon/routes/transfer.d.ts +4 -0
  129. package/dist/daemon/routes/transfer.d.ts.map +1 -0
  130. package/dist/daemon/routes/transfer.js +135 -0
  131. package/dist/daemon/routes/transfer.js.map +1 -0
  132. package/dist/daemon/server.d.ts +8 -0
  133. package/dist/daemon/server.d.ts.map +1 -0
  134. package/dist/daemon/server.js +170 -0
  135. package/dist/daemon/server.js.map +1 -0
  136. package/dist/daemon/services/ai-task-manager.d.ts +56 -0
  137. package/dist/daemon/services/ai-task-manager.d.ts.map +1 -0
  138. package/dist/daemon/services/ai-task-manager.js +491 -0
  139. package/dist/daemon/services/ai-task-manager.js.map +1 -0
  140. package/dist/daemon/services/audit-logger.d.ts +16 -0
  141. package/dist/daemon/services/audit-logger.d.ts.map +1 -0
  142. package/dist/daemon/services/audit-logger.js +23 -0
  143. package/dist/daemon/services/audit-logger.js.map +1 -0
  144. package/dist/daemon/services/context-store.d.ts +17 -0
  145. package/dist/daemon/services/context-store.d.ts.map +1 -0
  146. package/dist/daemon/services/context-store.js +97 -0
  147. package/dist/daemon/services/context-store.js.map +1 -0
  148. package/dist/daemon/services/job-manager.d.ts +19 -0
  149. package/dist/daemon/services/job-manager.d.ts.map +1 -0
  150. package/dist/daemon/services/job-manager.js +92 -0
  151. package/dist/daemon/services/job-manager.js.map +1 -0
  152. package/dist/daemon/services/tls-manager.d.ts +33 -0
  153. package/dist/daemon/services/tls-manager.d.ts.map +1 -0
  154. package/dist/daemon/services/tls-manager.js +114 -0
  155. package/dist/daemon/services/tls-manager.js.map +1 -0
  156. package/dist/daemon/utils/which.d.ts +2 -0
  157. package/dist/daemon/utils/which.d.ts.map +1 -0
  158. package/dist/daemon/utils/which.js +18 -0
  159. package/dist/daemon/utils/which.js.map +1 -0
  160. package/dist/dashboard/config.d.ts +8 -0
  161. package/dist/dashboard/config.d.ts.map +1 -0
  162. package/dist/dashboard/config.js +22 -0
  163. package/dist/dashboard/config.js.map +1 -0
  164. package/dist/dashboard/public/app.js +120 -0
  165. package/dist/dashboard/public/icon-192.png +0 -0
  166. package/dist/dashboard/public/icon-512.png +0 -0
  167. package/dist/dashboard/public/index.html +85 -0
  168. package/dist/dashboard/public/manifest.json +12 -0
  169. package/dist/dashboard/public/style.css +784 -0
  170. package/dist/dashboard/public/sw.js +31 -0
  171. package/dist/dashboard/public/views/ai-tasks.js +679 -0
  172. package/dist/dashboard/public/views/context.js +167 -0
  173. package/dist/dashboard/public/views/messages.js +263 -0
  174. package/dist/dashboard/public/views/overview.js +228 -0
  175. package/dist/dashboard/public/views/peers.js +136 -0
  176. package/dist/dashboard/public/views/terminal.js +153 -0
  177. package/dist/dashboard/routes/ai-tasks.d.ts +3 -0
  178. package/dist/dashboard/routes/ai-tasks.d.ts.map +1 -0
  179. package/dist/dashboard/routes/ai-tasks.js +193 -0
  180. package/dist/dashboard/routes/ai-tasks.js.map +1 -0
  181. package/dist/dashboard/routes/messages.d.ts +3 -0
  182. package/dist/dashboard/routes/messages.d.ts.map +1 -0
  183. package/dist/dashboard/routes/messages.js +137 -0
  184. package/dist/dashboard/routes/messages.js.map +1 -0
  185. package/dist/dashboard/routes/peer-utils.d.ts +17 -0
  186. package/dist/dashboard/routes/peer-utils.d.ts.map +1 -0
  187. package/dist/dashboard/routes/peer-utils.js +193 -0
  188. package/dist/dashboard/routes/peer-utils.js.map +1 -0
  189. package/dist/dashboard/routes/peers-all.d.ts +3 -0
  190. package/dist/dashboard/routes/peers-all.d.ts.map +1 -0
  191. package/dist/dashboard/routes/peers-all.js +8 -0
  192. package/dist/dashboard/routes/peers-all.js.map +1 -0
  193. package/dist/dashboard/routes/proxy.d.ts +3 -0
  194. package/dist/dashboard/routes/proxy.d.ts.map +1 -0
  195. package/dist/dashboard/routes/proxy.js +59 -0
  196. package/dist/dashboard/routes/proxy.js.map +1 -0
  197. package/dist/dashboard/routes/sessions.d.ts +3 -0
  198. package/dist/dashboard/routes/sessions.d.ts.map +1 -0
  199. package/dist/dashboard/routes/sessions.js +64 -0
  200. package/dist/dashboard/routes/sessions.js.map +1 -0
  201. package/dist/dashboard/routes/sse.d.ts +3 -0
  202. package/dist/dashboard/routes/sse.d.ts.map +1 -0
  203. package/dist/dashboard/routes/sse.js +49 -0
  204. package/dist/dashboard/routes/sse.js.map +1 -0
  205. package/dist/dashboard/routes/status.d.ts +3 -0
  206. package/dist/dashboard/routes/status.d.ts.map +1 -0
  207. package/dist/dashboard/routes/status.js +38 -0
  208. package/dist/dashboard/routes/status.js.map +1 -0
  209. package/dist/dashboard/server.d.ts +3 -0
  210. package/dist/dashboard/server.d.ts.map +1 -0
  211. package/dist/dashboard/server.js +77 -0
  212. package/dist/dashboard/server.js.map +1 -0
  213. package/dist/dashboard/session-manager.d.ts +17 -0
  214. package/dist/dashboard/session-manager.d.ts.map +1 -0
  215. package/dist/dashboard/session-manager.js +225 -0
  216. package/dist/dashboard/session-manager.js.map +1 -0
  217. package/dist/discovery/health-checker.d.ts +15 -0
  218. package/dist/discovery/health-checker.d.ts.map +1 -0
  219. package/dist/discovery/health-checker.js +47 -0
  220. package/dist/discovery/health-checker.js.map +1 -0
  221. package/dist/discovery/index.d.ts +4 -0
  222. package/dist/discovery/index.d.ts.map +1 -0
  223. package/dist/discovery/index.js +4 -0
  224. package/dist/discovery/index.js.map +1 -0
  225. package/dist/discovery/mdns.d.ts +21 -0
  226. package/dist/discovery/mdns.d.ts.map +1 -0
  227. package/dist/discovery/mdns.js +83 -0
  228. package/dist/discovery/mdns.js.map +1 -0
  229. package/dist/discovery/peer-registry.d.ts +18 -0
  230. package/dist/discovery/peer-registry.d.ts.map +1 -0
  231. package/dist/discovery/peer-registry.js +81 -0
  232. package/dist/discovery/peer-registry.js.map +1 -0
  233. package/dist/mcp-server/daemon-client.d.ts +69 -0
  234. package/dist/mcp-server/daemon-client.d.ts.map +1 -0
  235. package/dist/mcp-server/daemon-client.js +281 -0
  236. package/dist/mcp-server/daemon-client.js.map +1 -0
  237. package/dist/mcp-server/index.d.ts +3 -0
  238. package/dist/mcp-server/index.d.ts.map +1 -0
  239. package/dist/mcp-server/index.js +406 -0
  240. package/dist/mcp-server/index.js.map +1 -0
  241. package/dist/protocol/constants.d.ts +66 -0
  242. package/dist/protocol/constants.d.ts.map +1 -0
  243. package/dist/protocol/constants.js +66 -0
  244. package/dist/protocol/constants.js.map +1 -0
  245. package/dist/protocol/errors.d.ts +47 -0
  246. package/dist/protocol/errors.d.ts.map +1 -0
  247. package/dist/protocol/errors.js +62 -0
  248. package/dist/protocol/errors.js.map +1 -0
  249. package/dist/protocol/index.d.ts +5 -0
  250. package/dist/protocol/index.d.ts.map +1 -0
  251. package/dist/protocol/index.js +5 -0
  252. package/dist/protocol/index.js.map +1 -0
  253. package/dist/protocol/schemas.d.ts +209 -0
  254. package/dist/protocol/schemas.d.ts.map +1 -0
  255. package/dist/protocol/schemas.js +115 -0
  256. package/dist/protocol/schemas.js.map +1 -0
  257. package/dist/protocol/types.d.ts +302 -0
  258. package/dist/protocol/types.d.ts.map +1 -0
  259. package/dist/protocol/types.js +2 -0
  260. package/dist/protocol/types.js.map +1 -0
  261. package/package.json +50 -0
  262. 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,2 @@
1
+ export declare function which(command: string): Promise<string | null>;
2
+ //# sourceMappingURL=which.d.ts.map
@@ -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,8 @@
1
+ export interface DashboardConfig {
2
+ apiKey: string;
3
+ mainPort: number;
4
+ allowedKeys: Record<string, string>;
5
+ hostname?: string;
6
+ }
7
+ export declare function loadDashboardConfig(): Promise<DashboardConfig>;
8
+ //# sourceMappingURL=config.d.ts.map
@@ -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();