@interactive-inc/claude-funnel 0.10.0 → 0.15.1

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 (236) hide show
  1. package/README.md +106 -56
  2. package/dist/bin.js +557 -530
  3. package/dist/connectors/schedule.d.ts +2 -49
  4. package/dist/connectors/schedule.js +1 -1
  5. package/dist/connectors/slack.d.ts +4 -48
  6. package/dist/connectors/slack.js +1 -1
  7. package/dist/gateway/daemon.js +213 -211
  8. package/dist/index.d.ts +465 -173
  9. package/dist/index.js +692 -154
  10. package/dist/{schedule-connector-schema-CkuIQ0JQ.js → schedule-connector-schema-FxP7LPlx.js} +11 -0
  11. package/dist/{file-system-Co60LrmR.d.ts → schedule-listener-BPodvbld.d.ts} +56 -1
  12. package/dist/{slack-connector-schema-Cd22WiHB.js → slack-connector-schema-B4hsf3AY.js} +10 -1
  13. package/dist/slack-listener-CHj6uMY-.d.ts +74 -0
  14. package/package.json +2 -6
  15. package/schemas/funnel.schema.json +144 -0
  16. package/dist/slack-connector-schema-D7zAHN8k.d.ts +0 -15
  17. package/lib/bin.ts +0 -3
  18. package/lib/cli/factory.ts +0 -10
  19. package/lib/cli/index.ts +0 -85
  20. package/lib/cli/router/query-to-cli-args.ts +0 -20
  21. package/lib/cli/router/to-request.ts +0 -113
  22. package/lib/cli/router/validator.ts +0 -27
  23. package/lib/cli/routes/channels.$channel.connectors.$connector.rename.$newName.ts +0 -27
  24. package/lib/cli/routes/channels.$channel.connectors.$connector.request.ts +0 -40
  25. package/lib/cli/routes/channels.$channel.connectors.$connector.schedules.add.$id.ts +0 -41
  26. package/lib/cli/routes/channels.$channel.connectors.$connector.schedules.remove.$id.ts +0 -22
  27. package/lib/cli/routes/channels.$channel.connectors.$connector.schedules.ts +0 -23
  28. package/lib/cli/routes/channels.$channel.connectors.$connector.ts +0 -26
  29. package/lib/cli/routes/channels.$channel.connectors.add.$connector.ts +0 -92
  30. package/lib/cli/routes/channels.$channel.connectors.remove.$connector.ts +0 -22
  31. package/lib/cli/routes/channels.$channel.connectors.set.$connector.ts +0 -63
  32. package/lib/cli/routes/channels.$channel.connectors.ts +0 -26
  33. package/lib/cli/routes/channels.$channel.publish.ts +0 -52
  34. package/lib/cli/routes/channels.$channel.rename.$newName.ts +0 -22
  35. package/lib/cli/routes/channels.$channel.set.delivery.$mode.ts +0 -34
  36. package/lib/cli/routes/channels.$channel.ts +0 -34
  37. package/lib/cli/routes/channels.add.$channel.ts +0 -33
  38. package/lib/cli/routes/channels.remove.$channel.ts +0 -20
  39. package/lib/cli/routes/channels.ts +0 -39
  40. package/lib/cli/routes/claude.ts +0 -70
  41. package/lib/cli/routes/gateway.listeners.ts +0 -41
  42. package/lib/cli/routes/gateway.logs.ts +0 -123
  43. package/lib/cli/routes/gateway.restart.ts +0 -50
  44. package/lib/cli/routes/gateway.run.ts +0 -41
  45. package/lib/cli/routes/gateway.start.ts +0 -50
  46. package/lib/cli/routes/gateway.status.ts +0 -19
  47. package/lib/cli/routes/gateway.stop.ts +0 -32
  48. package/lib/cli/routes/gateway.ts +0 -55
  49. package/lib/cli/routes/index.ts +0 -219
  50. package/lib/cli/routes/profiles.$profile.as-default.ts +0 -22
  51. package/lib/cli/routes/profiles.$profile.rename.$newName.ts +0 -22
  52. package/lib/cli/routes/profiles.$profile.run.ts +0 -36
  53. package/lib/cli/routes/profiles.add.$profile.ts +0 -49
  54. package/lib/cli/routes/profiles.remove.$profile.ts +0 -20
  55. package/lib/cli/routes/profiles.set.$profile.ts +0 -45
  56. package/lib/cli/routes/profiles.ts +0 -40
  57. package/lib/cli/routes/status.ts +0 -93
  58. package/lib/cli/routes/update.ts +0 -27
  59. package/lib/connectors/connector-adapter.ts +0 -9
  60. package/lib/connectors/connector-config-schema.ts +0 -16
  61. package/lib/connectors/connector-factory.ts +0 -94
  62. package/lib/connectors/connector-listener.ts +0 -20
  63. package/lib/connectors/discord-adapter.ts +0 -51
  64. package/lib/connectors/discord-connector-schema.ts +0 -12
  65. package/lib/connectors/discord-event-processor.ts +0 -48
  66. package/lib/connectors/discord-listener.ts +0 -111
  67. package/lib/connectors/discord.ts +0 -4
  68. package/lib/connectors/gh-adapter.ts +0 -48
  69. package/lib/connectors/gh-connector-schema.ts +0 -12
  70. package/lib/connectors/gh-listener.ts +0 -137
  71. package/lib/connectors/gh.ts +0 -3
  72. package/lib/connectors/match-cron.ts +0 -78
  73. package/lib/connectors/schedule-connector-schema.ts +0 -33
  74. package/lib/connectors/schedule-listener.ts +0 -207
  75. package/lib/connectors/schedule-state-store.ts +0 -54
  76. package/lib/connectors/schedule.ts +0 -4
  77. package/lib/connectors/slack-adapter.ts +0 -36
  78. package/lib/connectors/slack-connector-schema.ts +0 -13
  79. package/lib/connectors/slack-event-processor.ts +0 -97
  80. package/lib/connectors/slack-listener.ts +0 -97
  81. package/lib/connectors/slack.ts +0 -4
  82. package/lib/engine/channels/channels.ts +0 -520
  83. package/lib/engine/claude/claude.ts +0 -205
  84. package/lib/engine/claude/gateway-controller.ts +0 -4
  85. package/lib/engine/fs/file-system.ts +0 -23
  86. package/lib/engine/fs/memory-file-system.ts +0 -102
  87. package/lib/engine/fs/node-file-system.ts +0 -68
  88. package/lib/engine/http/http-client.ts +0 -17
  89. package/lib/engine/http/memory-http-client.ts +0 -36
  90. package/lib/engine/http/node-http-client.ts +0 -23
  91. package/lib/engine/id/id-generator.ts +0 -7
  92. package/lib/engine/id/memory-id-generator.ts +0 -20
  93. package/lib/engine/id/node-id-generator.ts +0 -7
  94. package/lib/engine/logger/logger.ts +0 -11
  95. package/lib/engine/logger/memory-logger.ts +0 -28
  96. package/lib/engine/logger/node-logger.ts +0 -49
  97. package/lib/engine/logger/noop-logger.ts +0 -9
  98. package/lib/engine/mcp/channel-server.ts +0 -123
  99. package/lib/engine/mcp/channel-subscriber.ts +0 -82
  100. package/lib/engine/mcp/mcp.ts +0 -126
  101. package/lib/engine/mcp/read-channel-connectors.ts +0 -34
  102. package/lib/engine/mcp/read-gateway-token.ts +0 -16
  103. package/lib/engine/mcp/usage-hint-for-type.ts +0 -15
  104. package/lib/engine/process/memory-process-runner.ts +0 -88
  105. package/lib/engine/process/node-process-runner.ts +0 -91
  106. package/lib/engine/process/process-runner.ts +0 -33
  107. package/lib/engine/profiles/profile-channel-checker.ts +0 -7
  108. package/lib/engine/profiles/profiles.ts +0 -126
  109. package/lib/engine/settings/mock-settings-reader.ts +0 -27
  110. package/lib/engine/settings/settings-reader.ts +0 -6
  111. package/lib/engine/settings/settings-schema.ts +0 -48
  112. package/lib/engine/settings/settings-store.ts +0 -110
  113. package/lib/engine/time/clock.ts +0 -15
  114. package/lib/engine/time/memory-clock.ts +0 -26
  115. package/lib/engine/time/node-clock.ts +0 -7
  116. package/lib/funnel.ts +0 -294
  117. package/lib/gateway/auth-middleware.ts +0 -44
  118. package/lib/gateway/broadcaster.ts +0 -319
  119. package/lib/gateway/channel-publisher.ts +0 -67
  120. package/lib/gateway/daemon.ts +0 -47
  121. package/lib/gateway/factory.ts +0 -10
  122. package/lib/gateway/funnel-event-store.ts +0 -155
  123. package/lib/gateway/gateway-server.ts +0 -426
  124. package/lib/gateway/gateway-token.ts +0 -79
  125. package/lib/gateway/gateway.ts +0 -209
  126. package/lib/gateway/kill-competing-slack-gateways.ts +0 -56
  127. package/lib/gateway/listener-supervisor.ts +0 -339
  128. package/lib/gateway/listeners-client.ts +0 -128
  129. package/lib/gateway/publish-schema.ts +0 -27
  130. package/lib/gateway/resolve-daemon-script.ts +0 -26
  131. package/lib/gateway/routes/channels.connectors.call.ts +0 -39
  132. package/lib/gateway/routes/channels.publish.ts +0 -44
  133. package/lib/gateway/routes/health.ts +0 -13
  134. package/lib/gateway/routes/index.ts +0 -26
  135. package/lib/gateway/routes/listeners.list.ts +0 -6
  136. package/lib/gateway/routes/listeners.restart.ts +0 -15
  137. package/lib/gateway/routes/listeners.start.ts +0 -15
  138. package/lib/gateway/routes/listeners.stop.ts +0 -15
  139. package/lib/gateway/routes/route-deps.ts +0 -19
  140. package/lib/gateway/routes/status.ts +0 -15
  141. package/lib/gateway/routes/validator.ts +0 -17
  142. package/lib/index.ts +0 -67
  143. package/lib/logger/leuco-human-file-writer.ts +0 -65
  144. package/lib/logger/leuco-human-logger.ts +0 -98
  145. package/lib/logger/leuco-human-record.ts +0 -16
  146. package/lib/logger/leuco-human-stdout-writer.ts +0 -26
  147. package/lib/logger/leuco-human-writer.ts +0 -14
  148. package/lib/logger/leuco-logger-memory-sink.ts +0 -67
  149. package/lib/logger/leuco-logger-record.ts +0 -13
  150. package/lib/logger/leuco-logger-sink.ts +0 -33
  151. package/lib/logger/leuco-logger-sqlite-sink.ts +0 -355
  152. package/lib/logger/leuco-logger.ts +0 -135
  153. package/lib/tui/app.tsx +0 -357
  154. package/lib/tui/components/add-row.tsx +0 -18
  155. package/lib/tui/components/brand.tsx +0 -27
  156. package/lib/tui/components/card.tsx +0 -44
  157. package/lib/tui/components/detail-bar.tsx +0 -46
  158. package/lib/tui/components/editable-field.tsx +0 -33
  159. package/lib/tui/components/empty-state.tsx +0 -11
  160. package/lib/tui/components/gateway-status.tsx +0 -66
  161. package/lib/tui/components/keymap.tsx +0 -29
  162. package/lib/tui/components/menu-item.tsx +0 -73
  163. package/lib/tui/components/menu.tsx +0 -26
  164. package/lib/tui/components/panel-header.tsx +0 -22
  165. package/lib/tui/components/readonly-field.tsx +0 -18
  166. package/lib/tui/components/section-header.tsx +0 -25
  167. package/lib/tui/components/selection-accent.tsx +0 -32
  168. package/lib/tui/components/session-item.tsx +0 -33
  169. package/lib/tui/components/session-list.tsx +0 -33
  170. package/lib/tui/components/ui/hascii/accordion-item.tsx +0 -88
  171. package/lib/tui/components/ui/hascii/accordion.tsx +0 -96
  172. package/lib/tui/components/ui/hascii/alert-dialog.tsx +0 -43
  173. package/lib/tui/components/ui/hascii/badge.tsx +0 -51
  174. package/lib/tui/components/ui/hascii/breadcrumb.tsx +0 -58
  175. package/lib/tui/components/ui/hascii/button.tsx +0 -194
  176. package/lib/tui/components/ui/hascii/card-content.tsx +0 -14
  177. package/lib/tui/components/ui/hascii/card-description.tsx +0 -13
  178. package/lib/tui/components/ui/hascii/card-footer.tsx +0 -14
  179. package/lib/tui/components/ui/hascii/card-header.tsx +0 -14
  180. package/lib/tui/components/ui/hascii/card-title.tsx +0 -13
  181. package/lib/tui/components/ui/hascii/card.tsx +0 -27
  182. package/lib/tui/components/ui/hascii/checkbox.tsx +0 -65
  183. package/lib/tui/components/ui/hascii/command.tsx +0 -159
  184. package/lib/tui/components/ui/hascii/dialog-content.tsx +0 -14
  185. package/lib/tui/components/ui/hascii/dialog-description.tsx +0 -13
  186. package/lib/tui/components/ui/hascii/dialog-footer.tsx +0 -14
  187. package/lib/tui/components/ui/hascii/dialog-header.tsx +0 -14
  188. package/lib/tui/components/ui/hascii/dialog-title.tsx +0 -13
  189. package/lib/tui/components/ui/hascii/dialog.tsx +0 -27
  190. package/lib/tui/components/ui/hascii/file-tree.tsx +0 -142
  191. package/lib/tui/components/ui/hascii/focus-group.tsx +0 -62
  192. package/lib/tui/components/ui/hascii/form-item.tsx +0 -43
  193. package/lib/tui/components/ui/hascii/input-otp.tsx +0 -86
  194. package/lib/tui/components/ui/hascii/input.tsx +0 -130
  195. package/lib/tui/components/ui/hascii/pagination.tsx +0 -105
  196. package/lib/tui/components/ui/hascii/progress.tsx +0 -28
  197. package/lib/tui/components/ui/hascii/select.tsx +0 -131
  198. package/lib/tui/components/ui/hascii/separator.tsx +0 -35
  199. package/lib/tui/components/ui/hascii/sidebar-content.tsx +0 -23
  200. package/lib/tui/components/ui/hascii/sidebar-header.tsx +0 -14
  201. package/lib/tui/components/ui/hascii/sidebar-menu-item.tsx +0 -67
  202. package/lib/tui/components/ui/hascii/sidebar.tsx +0 -24
  203. package/lib/tui/components/ui/hascii/skeleton.tsx +0 -60
  204. package/lib/tui/components/ui/hascii/slider.tsx +0 -91
  205. package/lib/tui/components/ui/hascii/snackbar.tsx +0 -75
  206. package/lib/tui/components/ui/hascii/sparkline.tsx +0 -53
  207. package/lib/tui/components/ui/hascii/spinner.tsx +0 -47
  208. package/lib/tui/components/ui/hascii/stepper.tsx +0 -54
  209. package/lib/tui/components/ui/hascii/switch.tsx +0 -66
  210. package/lib/tui/components/ui/hascii/table.tsx +0 -95
  211. package/lib/tui/components/ui/hascii/tabs.tsx +0 -59
  212. package/lib/tui/components/ui/hascii/toggle-group-item.tsx +0 -45
  213. package/lib/tui/components/ui/hascii/toggle-group.tsx +0 -99
  214. package/lib/tui/components/ui/hascii/tree.tsx +0 -104
  215. package/lib/tui/components/view-shell.tsx +0 -44
  216. package/lib/tui/filter-input.tsx +0 -33
  217. package/lib/tui/hooks/hascii/use-pressable.ts +0 -54
  218. package/lib/tui/parse-comma-list.ts +0 -14
  219. package/lib/tui/profile-launcher.tsx +0 -61
  220. package/lib/tui/scrollbar-options.ts +0 -19
  221. package/lib/tui/sidebar.tsx +0 -50
  222. package/lib/tui/theme.ts +0 -40
  223. package/lib/tui/tui.tsx +0 -20
  224. package/lib/tui/types.ts +0 -38
  225. package/lib/tui/unique-name.ts +0 -18
  226. package/lib/tui/use-event-stream.ts +0 -133
  227. package/lib/tui/use-snapshot.ts +0 -99
  228. package/lib/tui/utils/hascii/form-item-context.tsx +0 -23
  229. package/lib/tui/utils/hascii/input-focus-context.tsx +0 -31
  230. package/lib/tui/utils/hascii/theme-context.tsx +0 -26
  231. package/lib/tui/utils/hascii/theme.ts +0 -176
  232. package/lib/tui/views/channels-view.tsx +0 -108
  233. package/lib/tui/views/connectors-view.tsx +0 -164
  234. package/lib/tui/views/events-view.tsx +0 -160
  235. package/lib/tui/views/listeners-view.tsx +0 -80
  236. package/lib/tui/views/profiles-view.tsx +0 -152
@@ -1,41 +0,0 @@
1
- import { HTTPException } from "hono/http-exception"
2
- import { z } from "zod"
3
- import { factory } from "@/cli/factory"
4
- import { zValidator } from "@/cli/router/validator"
5
-
6
- export const listenersHelp = `funnel gateway listeners — show running connector listeners
7
-
8
- usage: funnel gateway listeners
9
-
10
- Reads /listeners from the running gateway daemon and prints the live registry.
11
-
12
- examples:
13
- funnel gateway listeners`
14
-
15
- export const gatewayListenersHandler = factory.createHandlers(
16
- zValidator("query", z.object({}), listenersHelp),
17
- async (c) => {
18
- const funnel = c.var.funnel
19
- const result = await funnel.listeners.list()
20
-
21
- if (result.state === "offline") {
22
- throw new HTTPException(503, { message: "funnel gateway: not running" })
23
- }
24
-
25
- if (result.state === "error") {
26
- throw new HTTPException(503, { message: `funnel gateway: ${result.reason}` })
27
- }
28
-
29
- if (result.listeners.length === 0) {
30
- return c.text("funnel gateway: no running listeners")
31
- }
32
-
33
- const lines = result.listeners.map((entry) => {
34
- const health = entry.alive ? "alive" : "dead"
35
-
36
- return ` [${health.padEnd(5)}] ${entry.type.padEnd(8)} ${entry.name}`
37
- })
38
-
39
- return c.text(`funnel gateway: running listeners\n${lines.join("\n")}`)
40
- },
41
- )
@@ -1,123 +0,0 @@
1
- import { existsSync } from "node:fs"
2
- import { stringify } from "yaml"
3
- import { z } from "zod"
4
- import { factory } from "@/cli/factory"
5
- import { NodeFunnelLogger } from "@/engine/logger/node-logger"
6
- import { zValidator } from "@/cli/router/validator"
7
-
8
- export const logsHelp = `funnel gateway logs — tail diagnostic logs
9
-
10
- usage: funnel gateway logs [-n <N>]
11
-
12
- options:
13
- -n <N> number of trailing lines to show (default: 20)
14
-
15
- Tails /tmp/funnel/funnel.log (the daemon's diagnostic stream — gateway
16
- lifecycle, channel connect/disconnect, listener boot). Exit with SIGINT.
17
- Output is formatted as YAML.
18
-
19
- Domain events fanned out to WebSocket clients live in the SQLite event
20
- store (<logDir>/events.db); they are not shown here. Subscribe via the
21
- WS endpoint or query the store directly.
22
-
23
- examples:
24
- funnel gateway logs
25
- funnel gateway logs -n 100`
26
-
27
- const logger = new NodeFunnelLogger()
28
-
29
- type LogEntry = {
30
- time: string
31
- level: string
32
- message: string
33
- meta?: unknown
34
- }
35
-
36
- const tryParseJson = (line: string): unknown => {
37
- try {
38
- return JSON.parse(line)
39
- } catch {
40
- return null
41
- }
42
- }
43
-
44
- const isLogEntry = (value: unknown): value is LogEntry => {
45
- if (value === null || typeof value !== "object") return false
46
- if (!("time" in value) || typeof value.time !== "string") return false
47
- if (!("level" in value) || typeof value.level !== "string") return false
48
- if (!("message" in value) || typeof value.message !== "string") return false
49
-
50
- return true
51
- }
52
-
53
- export const gatewayLogsHandler = factory.createHandlers(
54
- zValidator(
55
- "query",
56
- z.object({
57
- n: z.string().optional(),
58
- }),
59
- logsHelp,
60
- ),
61
- async (c) => {
62
- const query = c.req.valid("query")
63
- const path = logger.file
64
-
65
- if (!path || !existsSync(path)) {
66
- return c.text("no logs")
67
- }
68
-
69
- const lineCount = query.n ? Number(query.n) : 20
70
-
71
- const tail = Bun.spawn(["tail", "-f", "-n", String(lineCount), path], {
72
- stdout: "pipe",
73
- stderr: "inherit",
74
- })
75
-
76
- const forward = (signal: "SIGINT" | "SIGTERM") => {
77
- tail.kill(signal)
78
- }
79
-
80
- process.on("SIGINT", () => forward("SIGINT"))
81
- process.on("SIGTERM", () => forward("SIGTERM"))
82
-
83
- const reader = tail.stdout.getReader()
84
- const decoder = new TextDecoder()
85
- let buffer = ""
86
-
87
- logger.info("gateway.logs tail start", { file: path })
88
-
89
- while (true) {
90
- const result = await reader.read()
91
-
92
- if (result.done) break
93
-
94
- buffer += decoder.decode(result.value, { stream: true })
95
-
96
- const lines = buffer.split("\n")
97
- buffer = lines.pop() ?? ""
98
-
99
- for (const line of lines) {
100
- if (!line.trim()) continue
101
-
102
- const parsed = tryParseJson(line)
103
-
104
- if (!isLogEntry(parsed)) {
105
- process.stdout.write(`${line}\n`)
106
- continue
107
- }
108
-
109
- const output = {
110
- time: parsed.time,
111
- level: parsed.level,
112
- message: parsed.message,
113
- ...(parsed.meta ? { meta: parsed.meta } : {}),
114
- }
115
-
116
- process.stdout.write(`---\n${stringify(output)}`)
117
- }
118
- }
119
-
120
- await tail.exited
121
- process.exit(0)
122
- },
123
- )
@@ -1,50 +0,0 @@
1
- import { HTTPException } from "hono/http-exception"
2
- import { z } from "zod"
3
- import { factory } from "@/cli/factory"
4
- import { zValidator } from "@/cli/router/validator"
5
-
6
- export const restartHelp = `funnel gateway restart — restart the gateway
7
-
8
- usage: funnel gateway restart [--no-caffeine]
9
-
10
- Stops the running process then starts it again in background.
11
- On macOS wraps with caffeinate -i by default. Use --no-caffeine to disable.
12
-
13
- examples:
14
- funnel gateway restart
15
- funnel gateway restart --no-caffeine`
16
-
17
- export const gatewayRestartHandler = factory.createHandlers(
18
- zValidator(
19
- "query",
20
- z.object({
21
- "no-caffeine": z.string().optional(),
22
- }),
23
- restartHelp,
24
- ),
25
- async (c) => {
26
- const query = c.req.valid("query")
27
- const funnel = c.var.funnel
28
-
29
- const result = await funnel.gateway.restart({
30
- caffeinate: query["no-caffeine"] !== "true",
31
- })
32
- const lines: string[] = []
33
-
34
- if (result.wasRunning) {
35
- lines.push(result.stopped ? "funnel gateway: stopped" : "funnel gateway: failed to stop")
36
- }
37
-
38
- if (result.stopped) {
39
- lines.push(result.started ? "funnel gateway: started" : "funnel gateway: failed to start")
40
- }
41
-
42
- const body = lines.join("\n")
43
-
44
- if (!result.ok) {
45
- throw new HTTPException(500, { message: body })
46
- }
47
-
48
- return c.text(body)
49
- },
50
- )
@@ -1,41 +0,0 @@
1
- import { z } from "zod"
2
- import { factory } from "@/cli/factory"
3
- import { zValidator } from "@/cli/router/validator"
4
- import { resolveDaemonScript } from "@/gateway/resolve-daemon-script"
5
-
6
- export const runHelp = `funnel gateway run — run the gateway in foreground
7
-
8
- usage: funnel gateway run [--no-caffeine]
9
-
10
- For developers. The process is tied to the current terminal and exits on SIGINT / SIGTERM.
11
- On macOS wraps with caffeinate -i by default. Use --no-caffeine to disable.
12
-
13
- For normal usage prefer funnel gateway start.
14
-
15
- examples:
16
- funnel gateway run
17
- funnel gateway run --no-caffeine`
18
-
19
- export const gatewayRunHandler = factory.createHandlers(
20
- zValidator(
21
- "query",
22
- z.object({
23
- "no-caffeine": z.string().optional(),
24
- }),
25
- runHelp,
26
- ),
27
- async (c) => {
28
- const query = c.req.valid("query")
29
- const funnel = c.var.funnel
30
-
31
- const gatewayScript = resolveDaemonScript()
32
- const useCaffeinate = query["no-caffeine"] !== "true" && process.platform === "darwin"
33
- const command = useCaffeinate
34
- ? ["caffeinate", "-i", "bun", gatewayScript]
35
- : ["bun", gatewayScript]
36
-
37
- const exitCode = await funnel.process.attach(command)
38
-
39
- process.exit(exitCode)
40
- },
41
- )
@@ -1,50 +0,0 @@
1
- import { HTTPException } from "hono/http-exception"
2
- import { z } from "zod"
3
- import { factory } from "@/cli/factory"
4
- import { zValidator } from "@/cli/router/validator"
5
-
6
- export const startHelp = `funnel gateway start — start the gateway in background
7
-
8
- usage: funnel gateway start [--no-caffeine]
9
-
10
- Daemonized with nohup, so it keeps running after the terminal is closed.
11
- On macOS wraps the process with caffeinate -i by default to prevent idle sleep.
12
- Use --no-caffeine to disable caffeinate.
13
-
14
- port: 9742 (override via FUNNEL_PORT)
15
- pid: ~/.funnel/gateway.pid
16
- log: /tmp/funnel/gateway.log
17
-
18
- examples:
19
- funnel gateway start
20
- funnel gateway start --no-caffeine`
21
-
22
- export const gatewayStartHandler = factory.createHandlers(
23
- zValidator(
24
- "query",
25
- z.object({
26
- "no-caffeine": z.string().optional(),
27
- }),
28
- startHelp,
29
- ),
30
- async (c) => {
31
- const query = c.req.valid("query")
32
- const funnel = c.var.funnel
33
-
34
- if (funnel.gateway.isRunning()) {
35
- const status = funnel.gateway.getStatus()
36
-
37
- return c.text(`funnel gateway: already running (pid ${status.pid})`)
38
- }
39
-
40
- const started = await funnel.gateway.start({
41
- caffeinate: query["no-caffeine"] !== "true",
42
- })
43
-
44
- if (!started) {
45
- throw new HTTPException(500, { message: "funnel gateway: failed to start" })
46
- }
47
-
48
- return c.text("funnel gateway: started")
49
- },
50
- )
@@ -1,19 +0,0 @@
1
- import { z } from "zod"
2
- import { factory } from "@/cli/factory"
3
- import { zValidator } from "@/cli/router/validator"
4
- import { renderGatewayStatus } from "@/cli/routes/gateway"
5
-
6
- export const statusHelp = `funnel gateway status — show gateway running status
7
-
8
- usage: funnel gateway status
9
-
10
- When running, prints PID, port, and connected channel count. When not running, exits with 503.
11
-
12
- examples:
13
- funnel gateway status
14
- funnel gateway`
15
-
16
- export const gatewayStatusHandler = factory.createHandlers(
17
- zValidator("query", z.object({}), statusHelp),
18
- renderGatewayStatus,
19
- )
@@ -1,32 +0,0 @@
1
- import { HTTPException } from "hono/http-exception"
2
- import { z } from "zod"
3
- import { factory } from "@/cli/factory"
4
- import { zValidator } from "@/cli/router/validator"
5
-
6
- export const stopHelp = `funnel gateway stop — stop the gateway
7
-
8
- usage: funnel gateway stop
9
-
10
- Terminates the process whose PID is stored in ~/.funnel/gateway.pid.
11
-
12
- examples:
13
- funnel gateway stop`
14
-
15
- export const gatewayStopHandler = factory.createHandlers(
16
- zValidator("query", z.object({}), stopHelp),
17
- async (c) => {
18
- const funnel = c.var.funnel
19
-
20
- if (!funnel.gateway.isRunning()) {
21
- return c.text("funnel gateway: no running process")
22
- }
23
-
24
- const stopped = await funnel.gateway.stop()
25
-
26
- if (!stopped) {
27
- throw new HTTPException(500, { message: "funnel gateway: failed to stop" })
28
- }
29
-
30
- return c.text("funnel gateway: stopped")
31
- },
32
- )
@@ -1,55 +0,0 @@
1
- import { HTTPException } from "hono/http-exception"
2
- import type { Context } from "hono"
3
- import { z } from "zod"
4
- import { factory } from "@/cli/factory"
5
- import type { Env } from "@/cli/factory"
6
- import { zValidator } from "@/cli/router/validator"
7
-
8
- export const groupHelp = `funnel gateway — manage the funnel daemon
9
-
10
- The gateway daemon hosts the WebSocket /ws (used by Claude MCP), the
11
- local web UI at /, and the listener supervisor that runs every
12
- connector. One daemon, one port (9742), one PID file.
13
-
14
- usage: funnel gateway [subcommand]
15
-
16
- subcommands:
17
- status show running status (default)
18
- start start in background
19
- stop stop
20
- restart stop then start
21
- run start in foreground (for developers)
22
- logs [-n <N>] show event logs
23
- listeners list running connector listeners (alive / dead)
24
-
25
- examples:
26
- funnel gateway check status
27
- funnel gateway restart restart`
28
-
29
- export const renderGatewayStatus = async (c: Context<Env>) => {
30
- const funnel = c.var.funnel
31
- const status = funnel.gateway.getStatus()
32
-
33
- if (!status.running) {
34
- throw new HTTPException(503, { message: "funnel gateway: not running" })
35
- }
36
-
37
- const res = await fetch(`http://localhost:${status.port}/health`).catch(() => null)
38
-
39
- if (!res) {
40
- return c.text(`funnel gateway: running (pid ${status.pid}) — health check failed`)
41
- }
42
-
43
- const health: unknown = await res.json()
44
- const clients =
45
- health !== null && typeof health === "object" && "clients" in health ? health.clients : 0
46
-
47
- return c.text(
48
- `funnel gateway: running (pid ${status.pid})\n port: ${status.port}\n clients: ${clients ?? 0}`,
49
- )
50
- }
51
-
52
- export const gatewayGroupHandler = factory.createHandlers(
53
- zValidator("query", z.object({}), groupHelp),
54
- renderGatewayStatus,
55
- )
@@ -1,219 +0,0 @@
1
- import { HTTPException } from "hono/http-exception"
2
- import { factory } from "@/cli/factory"
3
- import {
4
- addHelp as channelsAddHelp,
5
- channelsAddHandler,
6
- } from "@/cli/routes/channels.add.$channel"
7
- import { channelsConnectorsGroupHandler } from "@/cli/routes/channels.$channel.connectors"
8
- import {
9
- addHelp as channelsConnectorsAddHelp,
10
- channelsConnectorsAddHandler,
11
- } from "@/cli/routes/channels.$channel.connectors.add.$connector"
12
- import {
13
- channelsConnectorsRemoveHandler,
14
- removeHelp as channelsConnectorsRemoveHelp,
15
- } from "@/cli/routes/channels.$channel.connectors.remove.$connector"
16
- import {
17
- channelsConnectorsSetHandler,
18
- setHelp as channelsConnectorsSetHelp,
19
- } from "@/cli/routes/channels.$channel.connectors.set.$connector"
20
- import { channelsConnectorsShowHandler } from "@/cli/routes/channels.$channel.connectors.$connector"
21
- import {
22
- channelsConnectorsRenameHandler,
23
- renameHelp as channelsConnectorsRenameHelp,
24
- } from "@/cli/routes/channels.$channel.connectors.$connector.rename.$newName"
25
- import { channelsConnectorsRequestHandler } from "@/cli/routes/channels.$channel.connectors.$connector.request"
26
- import { channelsConnectorsSchedulesGroupHandler } from "@/cli/routes/channels.$channel.connectors.$connector.schedules"
27
- import {
28
- addHelp as channelsConnectorsSchedulesAddHelp,
29
- channelsConnectorsSchedulesAddHandler,
30
- } from "@/cli/routes/channels.$channel.connectors.$connector.schedules.add.$id"
31
- import {
32
- channelsConnectorsSchedulesRemoveHandler,
33
- removeHelp as channelsConnectorsSchedulesRemoveHelp,
34
- } from "@/cli/routes/channels.$channel.connectors.$connector.schedules.remove.$id"
35
- import {
36
- channelsPublishHandler,
37
- publishHelp as channelsPublishHelp,
38
- } from "@/cli/routes/channels.$channel.publish"
39
- import {
40
- channelsRemoveHandler,
41
- removeHelp as channelsRemoveHelp,
42
- } from "@/cli/routes/channels.remove.$channel"
43
- import {
44
- channelsRenameHandler,
45
- renameHelp as channelsRenameHelp,
46
- } from "@/cli/routes/channels.$channel.rename.$newName"
47
- import { channelsSetDeliveryHandler } from "@/cli/routes/channels.$channel.set.delivery.$mode"
48
- import { channelsShowHandler } from "@/cli/routes/channels.$channel"
49
- import { channelsGroupHandler } from "@/cli/routes/channels"
50
- import { claudeHandler } from "@/cli/routes/claude"
51
- import { gatewayGroupHandler } from "@/cli/routes/gateway"
52
- import { gatewayListenersHandler } from "@/cli/routes/gateway.listeners"
53
- import { gatewayLogsHandler } from "@/cli/routes/gateway.logs"
54
- import { gatewayRestartHandler } from "@/cli/routes/gateway.restart"
55
- import { gatewayRunHandler } from "@/cli/routes/gateway.run"
56
- import { gatewayStartHandler } from "@/cli/routes/gateway.start"
57
- import { gatewayStatusHandler } from "@/cli/routes/gateway.status"
58
- import { gatewayStopHandler } from "@/cli/routes/gateway.stop"
59
- import {
60
- addHelp as profilesAddHelp,
61
- profilesAddHandler,
62
- } from "@/cli/routes/profiles.add.$profile"
63
- import { profilesAsDefaultHandler } from "@/cli/routes/profiles.$profile.as-default"
64
- import {
65
- profilesRenameHandler,
66
- renameHelp as profilesRenameHelp,
67
- } from "@/cli/routes/profiles.$profile.rename.$newName"
68
- import { profilesLaunchHandler } from "@/cli/routes/profiles.$profile.run"
69
- import {
70
- profilesRemoveHandler,
71
- removeHelp as profilesRemoveHelp,
72
- } from "@/cli/routes/profiles.remove.$profile"
73
- import {
74
- profilesSetHandler,
75
- setHelp as profilesSetHelp,
76
- } from "@/cli/routes/profiles.set.$profile"
77
- import { profilesGroupHandler } from "@/cli/routes/profiles"
78
- import { statusHandler } from "@/cli/routes/status"
79
- import { updateHandler } from "@/cli/routes/update"
80
- import { Funnel } from "@/funnel"
81
-
82
- const helpRoute = (text: string) => factory.createHandlers((c) => c.text(text))
83
-
84
- /**
85
- * Build the CLI Hono app wired to a specific Funnel instance.
86
- * Exposed so library consumers can mount the same routes their `fnl` CLI
87
- * uses against a custom Funnel (e.g. one with sandboxed boundaries).
88
- *
89
- * All CLI verbs (`add` / `remove` / `set` / `rename` / `as-default` / `request`) map to POST in
90
- * to-request.ts and stay in the URL as a literal segment. Read paths (list / show / launch) keep GET.
91
- * Help shortcuts at parameterless URLs return the help text directly so `funnel <verb>` (no args) is
92
- * informative instead of 404.
93
- */
94
- export const createCliApp = (funnel: Funnel) => {
95
- const base = factory.createApp()
96
-
97
- base.use((c, next) => {
98
- c.set("funnel", funnel)
99
-
100
- return next()
101
- })
102
-
103
- base.onError((error, c) => {
104
- if (error instanceof HTTPException) {
105
- return c.text(`error: ${error.message}`, error.status)
106
- }
107
-
108
- return c.text(`error: ${error instanceof Error ? error.message : String(error)}`, 400)
109
- })
110
-
111
- return base
112
- .get("/claude", ...claudeHandler)
113
- .get("/channels", ...channelsGroupHandler)
114
- .post("/channels/add", ...helpRoute(channelsAddHelp))
115
- .post("/channels/add/:channel", ...channelsAddHandler)
116
- .post("/channels/remove", ...helpRoute(channelsRemoveHelp))
117
- .post("/channels/remove/:channel", ...channelsRemoveHandler)
118
- .post("/channels/rename/:channel/:newName", ...channelsRenameHandler)
119
- .post("/channels/:channel/rename/:newName", ...channelsRenameHandler)
120
- .post("/channels/rename", ...helpRoute(channelsRenameHelp))
121
- .post("/channels/:channel/rename", ...helpRoute(channelsRenameHelp))
122
- .post("/channels/:channel/set/delivery/:mode", ...channelsSetDeliveryHandler)
123
- .post("/channels/publish", ...helpRoute(channelsPublishHelp))
124
- .post("/channels/:channel/publish", ...channelsPublishHandler)
125
- .get("/channels/:channel", ...channelsShowHandler)
126
- .get("/channels/:channel/connectors", ...channelsConnectorsGroupHandler)
127
- .post(
128
- "/channels/:channel/connectors/add",
129
- ...helpRoute(channelsConnectorsAddHelp),
130
- )
131
- .post(
132
- "/channels/:channel/connectors/add/:connector",
133
- ...channelsConnectorsAddHandler,
134
- )
135
- .post(
136
- "/channels/:channel/connectors/remove",
137
- ...helpRoute(channelsConnectorsRemoveHelp),
138
- )
139
- .post(
140
- "/channels/:channel/connectors/remove/:connector",
141
- ...channelsConnectorsRemoveHandler,
142
- )
143
- .post(
144
- "/channels/:channel/connectors/set",
145
- ...helpRoute(channelsConnectorsSetHelp),
146
- )
147
- .post(
148
- "/channels/:channel/connectors/set/:connector",
149
- ...channelsConnectorsSetHandler,
150
- )
151
- .post(
152
- "/channels/:channel/connectors/rename/:connector/:newName",
153
- ...channelsConnectorsRenameHandler,
154
- )
155
- .post(
156
- "/channels/:channel/connectors/:connector/rename/:newName",
157
- ...channelsConnectorsRenameHandler,
158
- )
159
- .post(
160
- "/channels/:channel/connectors/rename",
161
- ...helpRoute(channelsConnectorsRenameHelp),
162
- )
163
- .post(
164
- "/channels/:channel/connectors/:connector/rename",
165
- ...helpRoute(channelsConnectorsRenameHelp),
166
- )
167
- .post(
168
- "/channels/:channel/connectors/:connector/request",
169
- ...channelsConnectorsRequestHandler,
170
- )
171
- .get("/channels/:channel/connectors/:connector", ...channelsConnectorsShowHandler)
172
- .get(
173
- "/channels/:channel/connectors/:connector/schedules",
174
- ...channelsConnectorsSchedulesGroupHandler,
175
- )
176
- .post(
177
- "/channels/:channel/connectors/:connector/schedules/add",
178
- ...helpRoute(channelsConnectorsSchedulesAddHelp),
179
- )
180
- .post(
181
- "/channels/:channel/connectors/:connector/schedules/add/:id",
182
- ...channelsConnectorsSchedulesAddHandler,
183
- )
184
- .post(
185
- "/channels/:channel/connectors/:connector/schedules/remove",
186
- ...helpRoute(channelsConnectorsSchedulesRemoveHelp),
187
- )
188
- .post(
189
- "/channels/:channel/connectors/:connector/schedules/remove/:id",
190
- ...channelsConnectorsSchedulesRemoveHandler,
191
- )
192
- .get("/profiles", ...profilesGroupHandler)
193
- .post("/profiles/add", ...helpRoute(profilesAddHelp))
194
- .post("/profiles/add/:profile", ...profilesAddHandler)
195
- .post("/profiles/set", ...helpRoute(profilesSetHelp))
196
- .post("/profiles/set/:profile", ...profilesSetHandler)
197
- .post("/profiles/remove", ...helpRoute(profilesRemoveHelp))
198
- .post("/profiles/remove/:profile", ...profilesRemoveHandler)
199
- .post("/profiles/rename/:profile/:newName", ...profilesRenameHandler)
200
- .post("/profiles/:profile/rename/:newName", ...profilesRenameHandler)
201
- .post("/profiles/rename", ...helpRoute(profilesRenameHelp))
202
- .post("/profiles/:profile/rename", ...helpRoute(profilesRenameHelp))
203
- .post("/profiles/:profile/as-default", ...profilesAsDefaultHandler)
204
- .get("/profiles/:profile/run", ...profilesLaunchHandler)
205
- .get("/profiles/:profile", ...profilesLaunchHandler)
206
- .get("/gateway", ...gatewayGroupHandler)
207
- .get("/gateway/status", ...gatewayStatusHandler)
208
- .get("/gateway/start", ...gatewayStartHandler)
209
- .get("/gateway/stop", ...gatewayStopHandler)
210
- .get("/gateway/restart", ...gatewayRestartHandler)
211
- .get("/gateway/run", ...gatewayRunHandler)
212
- .get("/gateway/logs", ...gatewayLogsHandler)
213
- .get("/gateway/listeners", ...gatewayListenersHandler)
214
- .get("/status", ...statusHandler)
215
- .get("/update", ...updateHandler)
216
- }
217
-
218
- /** CLI Hono app wired to a default `new Funnel()`. For embedding with a custom Funnel use `createCliApp`. */
219
- export const app = createCliApp(new Funnel())
@@ -1,22 +0,0 @@
1
- import { z } from "zod"
2
- import { factory } from "@/cli/factory"
3
- import { zValidator } from "@/cli/router/validator"
4
-
5
- export const asDefaultHelp = `funnel profiles <name> as-default — move profile to the front of the list
6
-
7
- usage: funnel profiles <name> as-default
8
-
9
- the first profile in the list is treated as the default for fnl claude.`
10
-
11
- export const profilesAsDefaultHandler = factory.createHandlers(
12
- zValidator("param", z.object({ profile: z.string() })),
13
- zValidator("query", z.object({}), asDefaultHelp),
14
- (c) => {
15
- const param = c.req.valid("param")
16
- const funnel = c.var.funnel
17
-
18
- funnel.profiles.asDefault(param.profile)
19
-
20
- return c.text(`profile "${param.profile}" is now the default`)
21
- },
22
- )
@@ -1,22 +0,0 @@
1
- import { z } from "zod"
2
- import { factory } from "@/cli/factory"
3
- import { zValidator } from "@/cli/router/validator"
4
-
5
- export const renameHelp = `funnel profiles rename — rename a profile
6
-
7
- usage:
8
- funnel profiles rename <old> <new>
9
- funnel profiles <old> rename <new>`
10
-
11
- export const profilesRenameHandler = factory.createHandlers(
12
- zValidator("param", z.object({ profile: z.string(), newName: z.string() })),
13
- zValidator("query", z.object({}), renameHelp),
14
- (c) => {
15
- const param = c.req.valid("param")
16
- const funnel = c.var.funnel
17
-
18
- funnel.profiles.rename(param.profile, param.newName)
19
-
20
- return c.text(`renamed profile "${param.profile}" to "${param.newName}"`)
21
- },
22
- )