@hellcoder/companion 0.96.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 (242) hide show
  1. package/bin/cli.ts +168 -0
  2. package/bin/ctl.ts +528 -0
  3. package/bin/generate-token.ts +28 -0
  4. package/dist/apple-touch-icon.png +0 -0
  5. package/dist/assets/AgentsPage-DCFhrJ28.js +13 -0
  6. package/dist/assets/CronManager-EGwLJONv.js +1 -0
  7. package/dist/assets/IntegrationsPage-CTMRnbQS.js +1 -0
  8. package/dist/assets/LinearOAuthSettingsPage-CgQFMIgr.js +1 -0
  9. package/dist/assets/LinearSettingsPage-C9nok1qi.js +1 -0
  10. package/dist/assets/Playground-BV3k0RbV.js +109 -0
  11. package/dist/assets/PromptsPage-CFojqNKP.js +4 -0
  12. package/dist/assets/RunsPage-DUJ1QUSa.js +1 -0
  13. package/dist/assets/SandboxManager-CrVQ-VU_.js +8 -0
  14. package/dist/assets/SettingsPage-D1fPCL19.js +1 -0
  15. package/dist/assets/TailscalePage-D06cyvyC.js +1 -0
  16. package/dist/assets/index-BhUa1e6X.css +1 -0
  17. package/dist/assets/index-DkqeP-R9.js +134 -0
  18. package/dist/assets/sw-register-BibwRdvC.js +1 -0
  19. package/dist/assets/workbox-window.prod.es5-BIl4cyR9.js +2 -0
  20. package/dist/favicon.svg +8 -0
  21. package/dist/fonts/MesloLGSNerdFontMono-Bold.woff2 +0 -0
  22. package/dist/fonts/MesloLGSNerdFontMono-Regular.woff2 +0 -0
  23. package/dist/icon-192.png +0 -0
  24. package/dist/icon-512.png +0 -0
  25. package/dist/index.html +20 -0
  26. package/dist/logo-codex.svg +14 -0
  27. package/dist/logo-docker.svg +4 -0
  28. package/dist/logo.svg +14 -0
  29. package/dist/manifest.json +24 -0
  30. package/dist/sw.js +2 -0
  31. package/package.json +104 -0
  32. package/server/agent-cron-migrator.test.ts +610 -0
  33. package/server/agent-cron-migrator.ts +85 -0
  34. package/server/agent-executor.test.ts +1108 -0
  35. package/server/agent-executor.ts +346 -0
  36. package/server/agent-store.test.ts +588 -0
  37. package/server/agent-store.ts +185 -0
  38. package/server/agent-types.ts +138 -0
  39. package/server/ai-validation-settings.test.ts +128 -0
  40. package/server/ai-validation-settings.ts +35 -0
  41. package/server/ai-validator.test.ts +387 -0
  42. package/server/ai-validator.ts +271 -0
  43. package/server/auth-manager.test.ts +83 -0
  44. package/server/auth-manager.ts +150 -0
  45. package/server/auto-namer.test.ts +252 -0
  46. package/server/auto-namer.ts +78 -0
  47. package/server/backend-adapter.test.ts +38 -0
  48. package/server/backend-adapter.ts +54 -0
  49. package/server/cache-headers.test.ts +98 -0
  50. package/server/cache-headers.ts +61 -0
  51. package/server/claude-adapter.test.ts +1363 -0
  52. package/server/claude-adapter.ts +889 -0
  53. package/server/claude-container-auth.test.ts +44 -0
  54. package/server/claude-container-auth.ts +30 -0
  55. package/server/claude-protocol-contract.test.ts +71 -0
  56. package/server/claude-protocol-drift.test.ts +78 -0
  57. package/server/claude-session-discovery.test.ts +132 -0
  58. package/server/claude-session-discovery.ts +157 -0
  59. package/server/claude-session-history.test.ts +158 -0
  60. package/server/claude-session-history.ts +410 -0
  61. package/server/cli-launcher.test.ts +1343 -0
  62. package/server/cli-launcher.ts +1298 -0
  63. package/server/cli.test.ts +16 -0
  64. package/server/codex-adapter.test.ts +5545 -0
  65. package/server/codex-adapter.ts +3062 -0
  66. package/server/codex-container-auth.test.ts +50 -0
  67. package/server/codex-container-auth.ts +24 -0
  68. package/server/codex-home.test.ts +61 -0
  69. package/server/codex-home.ts +26 -0
  70. package/server/codex-protocol-contract.test.ts +96 -0
  71. package/server/codex-protocol-drift.test.ts +123 -0
  72. package/server/codex-ws-proxy.cjs +226 -0
  73. package/server/commands-discovery.test.ts +179 -0
  74. package/server/commands-discovery.ts +81 -0
  75. package/server/constants.ts +7 -0
  76. package/server/container-manager.test.ts +1211 -0
  77. package/server/container-manager.ts +1053 -0
  78. package/server/cron-scheduler.test.ts +957 -0
  79. package/server/cron-scheduler.ts +243 -0
  80. package/server/cron-store.test.ts +422 -0
  81. package/server/cron-store.ts +148 -0
  82. package/server/cron-types.ts +63 -0
  83. package/server/env-manager.test.ts +268 -0
  84. package/server/env-manager.ts +161 -0
  85. package/server/event-bus-types.ts +64 -0
  86. package/server/event-bus.test.ts +244 -0
  87. package/server/event-bus.ts +124 -0
  88. package/server/execution-store.test.ts +307 -0
  89. package/server/execution-store.ts +170 -0
  90. package/server/fs-utils.ts +15 -0
  91. package/server/git-utils.test.ts +938 -0
  92. package/server/git-utils.ts +421 -0
  93. package/server/github-pr.test.ts +498 -0
  94. package/server/github-pr.ts +379 -0
  95. package/server/image-pull-manager.test.ts +303 -0
  96. package/server/image-pull-manager.ts +279 -0
  97. package/server/index.ts +396 -0
  98. package/server/linear-agent-bridge.test.ts +1157 -0
  99. package/server/linear-agent-bridge.ts +629 -0
  100. package/server/linear-agent.test.ts +473 -0
  101. package/server/linear-agent.ts +479 -0
  102. package/server/linear-cache.test.ts +136 -0
  103. package/server/linear-cache.ts +113 -0
  104. package/server/linear-connections.test.ts +350 -0
  105. package/server/linear-connections.ts +231 -0
  106. package/server/linear-credential-migration.test.ts +337 -0
  107. package/server/linear-credential-migration.ts +63 -0
  108. package/server/linear-oauth-connections-migration.test.ts +268 -0
  109. package/server/linear-oauth-connections.test.ts +365 -0
  110. package/server/linear-oauth-connections.ts +294 -0
  111. package/server/linear-project-manager.test.ts +162 -0
  112. package/server/linear-project-manager.ts +111 -0
  113. package/server/linear-prompt-builder.test.ts +74 -0
  114. package/server/linear-prompt-builder.ts +61 -0
  115. package/server/linear-staging.test.ts +276 -0
  116. package/server/linear-staging.ts +142 -0
  117. package/server/logger.test.ts +393 -0
  118. package/server/logger.ts +259 -0
  119. package/server/metrics-collector.test.ts +413 -0
  120. package/server/metrics-collector.ts +350 -0
  121. package/server/metrics-types.ts +108 -0
  122. package/server/middleware/managed-auth.test.ts +264 -0
  123. package/server/middleware/managed-auth.ts +195 -0
  124. package/server/novnc-proxy.test.ts +333 -0
  125. package/server/novnc-proxy.ts +99 -0
  126. package/server/path-resolver.test.ts +552 -0
  127. package/server/path-resolver.ts +186 -0
  128. package/server/paths.test.ts +31 -0
  129. package/server/paths.ts +11 -0
  130. package/server/pr-poller.test.ts +191 -0
  131. package/server/pr-poller.ts +162 -0
  132. package/server/prompt-manager.test.ts +211 -0
  133. package/server/prompt-manager.ts +211 -0
  134. package/server/protocol/claude-upstream/README.md +19 -0
  135. package/server/protocol/claude-upstream/sdk.d.ts.txt +1943 -0
  136. package/server/protocol/codex-upstream/ClientNotification.ts.txt +5 -0
  137. package/server/protocol/codex-upstream/ClientRequest.ts.txt +60 -0
  138. package/server/protocol/codex-upstream/README.md +18 -0
  139. package/server/protocol/codex-upstream/ServerNotification.ts.txt +41 -0
  140. package/server/protocol/codex-upstream/ServerRequest.ts.txt +16 -0
  141. package/server/protocol/codex-upstream/v2/DynamicToolCallParams.ts.txt +6 -0
  142. package/server/protocol/codex-upstream/v2/DynamicToolCallResponse.ts.txt +6 -0
  143. package/server/protocol-monitor.ts +50 -0
  144. package/server/recorder.test.ts +454 -0
  145. package/server/recorder.ts +374 -0
  146. package/server/recording-hub/compat-validator.test.ts +150 -0
  147. package/server/recording-hub/compat-validator.ts +284 -0
  148. package/server/recording-hub/diagnostics.test.ts +140 -0
  149. package/server/recording-hub/diagnostics.ts +299 -0
  150. package/server/recording-hub/hub-config.test.ts +44 -0
  151. package/server/recording-hub/hub-config.ts +19 -0
  152. package/server/recording-hub/hub-routes.test.ts +417 -0
  153. package/server/recording-hub/hub-routes.ts +236 -0
  154. package/server/recording-hub/hub-store.test.ts +262 -0
  155. package/server/recording-hub/hub-store.ts +265 -0
  156. package/server/recording-hub/replay-adapter.test.ts +294 -0
  157. package/server/recording-hub/replay-adapter.ts +207 -0
  158. package/server/relay-client.test.ts +337 -0
  159. package/server/relay-client.ts +320 -0
  160. package/server/replay.test.ts +200 -0
  161. package/server/replay.ts +78 -0
  162. package/server/routes/agent-routes.test.ts +1400 -0
  163. package/server/routes/agent-routes.ts +409 -0
  164. package/server/routes/cron-routes.test.ts +881 -0
  165. package/server/routes/cron-routes.ts +103 -0
  166. package/server/routes/env-routes.test.ts +383 -0
  167. package/server/routes/env-routes.ts +95 -0
  168. package/server/routes/fs-routes.test.ts +1198 -0
  169. package/server/routes/fs-routes.ts +605 -0
  170. package/server/routes/git-routes.test.ts +813 -0
  171. package/server/routes/git-routes.ts +97 -0
  172. package/server/routes/linear-agent-routes.test.ts +721 -0
  173. package/server/routes/linear-agent-routes.ts +304 -0
  174. package/server/routes/linear-connection-routes.test.ts +927 -0
  175. package/server/routes/linear-connection-routes.ts +244 -0
  176. package/server/routes/linear-oauth-connection-routes.test.ts +406 -0
  177. package/server/routes/linear-oauth-connection-routes.ts +129 -0
  178. package/server/routes/linear-routes.test.ts +1510 -0
  179. package/server/routes/linear-routes.ts +953 -0
  180. package/server/routes/metrics-routes.test.ts +103 -0
  181. package/server/routes/metrics-routes.ts +13 -0
  182. package/server/routes/prompt-routes.ts +67 -0
  183. package/server/routes/sandbox-routes.test.ts +513 -0
  184. package/server/routes/sandbox-routes.ts +127 -0
  185. package/server/routes/settings-routes.ts +270 -0
  186. package/server/routes/skills-routes.test.ts +690 -0
  187. package/server/routes/skills-routes.ts +100 -0
  188. package/server/routes/system-routes.test.ts +637 -0
  189. package/server/routes/system-routes.ts +228 -0
  190. package/server/routes/tailscale-routes.test.ts +176 -0
  191. package/server/routes/tailscale-routes.ts +22 -0
  192. package/server/routes.test.ts +4655 -0
  193. package/server/routes.ts +1277 -0
  194. package/server/sandbox-manager.test.ts +378 -0
  195. package/server/sandbox-manager.ts +168 -0
  196. package/server/service.test.ts +1419 -0
  197. package/server/service.ts +718 -0
  198. package/server/session-creation-service.test.ts +661 -0
  199. package/server/session-creation-service.ts +473 -0
  200. package/server/session-git-info.ts +104 -0
  201. package/server/session-linear-issues.test.ts +118 -0
  202. package/server/session-linear-issues.ts +88 -0
  203. package/server/session-names.test.ts +94 -0
  204. package/server/session-names.ts +67 -0
  205. package/server/session-orchestrator.test.ts +1784 -0
  206. package/server/session-orchestrator.ts +973 -0
  207. package/server/session-state-machine.test.ts +606 -0
  208. package/server/session-state-machine.ts +207 -0
  209. package/server/session-store.test.ts +290 -0
  210. package/server/session-store.ts +146 -0
  211. package/server/session-types.ts +509 -0
  212. package/server/settings-manager.test.ts +275 -0
  213. package/server/settings-manager.ts +173 -0
  214. package/server/tailscale-manager.test.ts +553 -0
  215. package/server/tailscale-manager.ts +451 -0
  216. package/server/terminal-manager.ts +240 -0
  217. package/server/update-checker.test.ts +306 -0
  218. package/server/update-checker.ts +197 -0
  219. package/server/usage-limits.test.ts +536 -0
  220. package/server/usage-limits.ts +225 -0
  221. package/server/worktree-tracker.test.ts +243 -0
  222. package/server/worktree-tracker.ts +84 -0
  223. package/server/ws-auth.test.ts +59 -0
  224. package/server/ws-auth.ts +41 -0
  225. package/server/ws-bridge-browser-ingest.test.ts +272 -0
  226. package/server/ws-bridge-browser-ingest.ts +72 -0
  227. package/server/ws-bridge-browser.ts +112 -0
  228. package/server/ws-bridge-cli-ingest.test.ts +302 -0
  229. package/server/ws-bridge-cli-ingest.ts +81 -0
  230. package/server/ws-bridge-codex.test.ts +1837 -0
  231. package/server/ws-bridge-codex.ts +266 -0
  232. package/server/ws-bridge-controls.test.ts +124 -0
  233. package/server/ws-bridge-controls.ts +20 -0
  234. package/server/ws-bridge-persist.test.ts +296 -0
  235. package/server/ws-bridge-persist.ts +66 -0
  236. package/server/ws-bridge-publish.test.ts +234 -0
  237. package/server/ws-bridge-publish.ts +79 -0
  238. package/server/ws-bridge-replay.test.ts +44 -0
  239. package/server/ws-bridge-replay.ts +61 -0
  240. package/server/ws-bridge-types.ts +106 -0
  241. package/server/ws-bridge.test.ts +4777 -0
  242. package/server/ws-bridge.ts +1279 -0
@@ -0,0 +1,103 @@
1
+ import { describe, it, expect, vi, beforeEach } from "vitest";
2
+ import { Hono } from "hono";
3
+ import { registerMetricsRoutes } from "./metrics-routes.js";
4
+
5
+ // Mock the singleton so tests don't pollute each other
6
+ vi.mock("../metrics-collector.js", () => {
7
+ const mockSnapshot = {
8
+ serverUptimeMs: 60_000,
9
+ snapshotAt: 1710500000000,
10
+ counters: {
11
+ sessionsCreated: { claude: 3 },
12
+ sessionsTerminated: { "0": 1 },
13
+ autoRelaunches: { attempted: 1, succeeded: 1, exhausted: 0 },
14
+ messagesProcessed: { assistant: 10 },
15
+ permissionRequests: {
16
+ total: 5,
17
+ autoApproved: 2,
18
+ autoDenied: 0,
19
+ userApproved: 3,
20
+ userDenied: 0,
21
+ },
22
+ errors: {},
23
+ stateTransitions: { "starting→initializing": 3 },
24
+ wsConnections: { cliOpened: 3, cliClosed: 1, browserOpened: 5, browserClosed: 2 },
25
+ },
26
+ gauges: {
27
+ activeSessions: { ready: 2 },
28
+ totalActiveSessions: 2,
29
+ connectedBrowsers: 3,
30
+ totalPendingMessages: 0,
31
+ totalEventBufferSize: 4,
32
+ totalHistoryMessages: 100,
33
+ memory: { rss: 100_000_000, heapUsed: 50_000_000, heapTotal: 70_000_000, external: 5_000_000 },
34
+ },
35
+ histograms: {
36
+ sessionInitTimeMs: { count: 3, sum: 9000, min: 2000, max: 4000, avg: 3000, p50Bucket: 2500, p95Bucket: 5000, p99Bucket: 5000, buckets: {} },
37
+ turnDurationMs: { count: 10, sum: 50000, min: 1000, max: 15000, avg: 5000, p50Bucket: 5000, p95Bucket: 10000, p99Bucket: 30000, buckets: {} },
38
+ permissionDurationMs: { count: 5, sum: 15000, min: 500, max: 8000, avg: 3000, p50Bucket: 2500, p95Bucket: 10000, p99Bucket: 10000, buckets: {} },
39
+ },
40
+ };
41
+
42
+ return {
43
+ metricsCollector: {
44
+ getSnapshot: vi.fn(() => mockSnapshot),
45
+ },
46
+ };
47
+ });
48
+
49
+ describe("GET /metrics", () => {
50
+ let app: Hono;
51
+
52
+ beforeEach(() => {
53
+ app = new Hono();
54
+ const mockGaugeProvider = {
55
+ getSessionMemoryStats: vi.fn(() => []),
56
+ getSessionPhases: vi.fn(() => new Map()),
57
+ };
58
+ registerMetricsRoutes(app, { gaugeProvider: mockGaugeProvider });
59
+ });
60
+
61
+ it("returns 200 with valid JSON", async () => {
62
+ const res = await app.request("/metrics");
63
+ expect(res.status).toBe(200);
64
+
65
+ const json = await res.json();
66
+ expect(json).toBeDefined();
67
+ });
68
+
69
+ it("response contains expected top-level keys", async () => {
70
+ const res = await app.request("/metrics");
71
+ const json = await res.json();
72
+
73
+ expect(json.serverUptimeMs).toBeDefined();
74
+ expect(json.snapshotAt).toBeDefined();
75
+ expect(json.counters).toBeDefined();
76
+ expect(json.gauges).toBeDefined();
77
+ expect(json.histograms).toBeDefined();
78
+ });
79
+
80
+ it("counters contain sessionsCreated", async () => {
81
+ const res = await app.request("/metrics");
82
+ const json = await res.json();
83
+
84
+ expect(json.counters.sessionsCreated).toEqual({ claude: 3 });
85
+ });
86
+
87
+ it("gauges contain memory info", async () => {
88
+ const res = await app.request("/metrics");
89
+ const json = await res.json();
90
+
91
+ expect(json.gauges.memory.rss).toBeGreaterThan(0);
92
+ expect(json.gauges.memory.heapUsed).toBeGreaterThan(0);
93
+ });
94
+
95
+ it("histograms contain session init time", async () => {
96
+ const res = await app.request("/metrics");
97
+ const json = await res.json();
98
+
99
+ expect(json.histograms.sessionInitTimeMs.count).toBe(3);
100
+ expect(json.histograms.turnDurationMs.count).toBe(10);
101
+ expect(json.histograms.permissionDurationMs.count).toBe(5);
102
+ });
103
+ });
@@ -0,0 +1,13 @@
1
+ import type { Hono } from "hono";
2
+ import type { GaugeDataProvider } from "../metrics-collector.js";
3
+ import { metricsCollector } from "../metrics-collector.js";
4
+
5
+ export function registerMetricsRoutes(
6
+ api: Hono,
7
+ deps: { gaugeProvider: GaugeDataProvider },
8
+ ): void {
9
+ api.get("/metrics", (c) => {
10
+ const snapshot = metricsCollector.getSnapshot(deps.gaugeProvider);
11
+ return c.json(snapshot);
12
+ });
13
+ }
@@ -0,0 +1,67 @@
1
+ import type { Hono } from "hono";
2
+ import * as promptManager from "../prompt-manager.js";
3
+
4
+ function sanitizePaths(value: unknown): string[] | undefined {
5
+ if (!Array.isArray(value)) return undefined;
6
+ return value.filter((p): p is string => typeof p === "string");
7
+ }
8
+
9
+ export function registerPromptRoutes(api: Hono): void {
10
+ api.get("/prompts", (c) => {
11
+ try {
12
+ const cwd = c.req.query("cwd");
13
+ const scope = c.req.query("scope");
14
+ const normalizedScope =
15
+ scope === "global" || scope === "project" || scope === "all"
16
+ ? scope
17
+ : undefined;
18
+ return c.json(promptManager.listPrompts({ cwd, scope: normalizedScope }));
19
+ } catch (e: unknown) {
20
+ return c.json({ error: e instanceof Error ? e.message : String(e) }, 500);
21
+ }
22
+ });
23
+
24
+ api.get("/prompts/:id", (c) => {
25
+ const prompt = promptManager.getPrompt(c.req.param("id"));
26
+ if (!prompt) return c.json({ error: "Prompt not found" }, 404);
27
+ return c.json(prompt);
28
+ });
29
+
30
+ api.post("/prompts", async (c) => {
31
+ const body = await c.req.json().catch(() => ({}));
32
+ try {
33
+ const prompt = promptManager.createPrompt(
34
+ String(body.title || body.name || ""),
35
+ String(body.content || ""),
36
+ body.scope,
37
+ body.cwd ?? body.projectPath,
38
+ sanitizePaths(body.projectPaths),
39
+ );
40
+ return c.json(prompt, 201);
41
+ } catch (e: unknown) {
42
+ return c.json({ error: e instanceof Error ? e.message : String(e) }, 400);
43
+ }
44
+ });
45
+
46
+ api.put("/prompts/:id", async (c) => {
47
+ const body = await c.req.json().catch(() => ({}));
48
+ try {
49
+ const prompt = promptManager.updatePrompt(c.req.param("id"), {
50
+ name: body.title ?? body.name,
51
+ content: body.content,
52
+ scope: body.scope,
53
+ projectPaths: sanitizePaths(body.projectPaths),
54
+ });
55
+ if (!prompt) return c.json({ error: "Prompt not found" }, 404);
56
+ return c.json(prompt);
57
+ } catch (e: unknown) {
58
+ return c.json({ error: e instanceof Error ? e.message : String(e) }, 400);
59
+ }
60
+ });
61
+
62
+ api.delete("/prompts/:id", (c) => {
63
+ const deleted = promptManager.deletePrompt(c.req.param("id"));
64
+ if (!deleted) return c.json({ error: "Prompt not found" }, 404);
65
+ return c.json({ ok: true });
66
+ });
67
+ }