uxnan-bridge 0.0.1-alpha.20260621

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 (255) hide show
  1. package/README.md +150 -0
  2. package/dist/src/account-status.d.ts +13 -0
  3. package/dist/src/account-status.js +78 -0
  4. package/dist/src/account-status.js.map +1 -0
  5. package/dist/src/adapters/base-adapter.d.ts +18 -0
  6. package/dist/src/adapters/base-adapter.js +15 -0
  7. package/dist/src/adapters/base-adapter.js.map +1 -0
  8. package/dist/src/adapters/claude-adapter.d.ts +102 -0
  9. package/dist/src/adapters/claude-adapter.js +486 -0
  10. package/dist/src/adapters/claude-adapter.js.map +1 -0
  11. package/dist/src/adapters/claude-tools.d.ts +25 -0
  12. package/dist/src/adapters/claude-tools.js +146 -0
  13. package/dist/src/adapters/claude-tools.js.map +1 -0
  14. package/dist/src/adapters/codex-adapter.d.ts +116 -0
  15. package/dist/src/adapters/codex-adapter.js +912 -0
  16. package/dist/src/adapters/codex-adapter.js.map +1 -0
  17. package/dist/src/adapters/codex-app-server.d.ts +74 -0
  18. package/dist/src/adapters/codex-app-server.js +225 -0
  19. package/dist/src/adapters/codex-app-server.js.map +1 -0
  20. package/dist/src/adapters/codex-approval.d.ts +88 -0
  21. package/dist/src/adapters/codex-approval.js +160 -0
  22. package/dist/src/adapters/codex-approval.js.map +1 -0
  23. package/dist/src/adapters/codex-tools.d.ts +18 -0
  24. package/dist/src/adapters/codex-tools.js +106 -0
  25. package/dist/src/adapters/codex-tools.js.map +1 -0
  26. package/dist/src/adapters/content-blocks.d.ts +68 -0
  27. package/dist/src/adapters/content-blocks.js +205 -0
  28. package/dist/src/adapters/content-blocks.js.map +1 -0
  29. package/dist/src/adapters/echo-agent-adapter.d.ts +23 -0
  30. package/dist/src/adapters/echo-agent-adapter.js +72 -0
  31. package/dist/src/adapters/echo-agent-adapter.js.map +1 -0
  32. package/dist/src/adapters/gemini-adapter.d.ts +87 -0
  33. package/dist/src/adapters/gemini-adapter.js +594 -0
  34. package/dist/src/adapters/gemini-adapter.js.map +1 -0
  35. package/dist/src/adapters/gemini-tools.d.ts +4 -0
  36. package/dist/src/adapters/gemini-tools.js +48 -0
  37. package/dist/src/adapters/gemini-tools.js.map +1 -0
  38. package/dist/src/adapters/opencode-adapter.d.ts +74 -0
  39. package/dist/src/adapters/opencode-adapter.js +418 -0
  40. package/dist/src/adapters/opencode-adapter.js.map +1 -0
  41. package/dist/src/adapters/opencode-tools.d.ts +2 -0
  42. package/dist/src/adapters/opencode-tools.js +41 -0
  43. package/dist/src/adapters/opencode-tools.js.map +1 -0
  44. package/dist/src/adapters/pi-adapter.d.ts +92 -0
  45. package/dist/src/adapters/pi-adapter.js +467 -0
  46. package/dist/src/adapters/pi-adapter.js.map +1 -0
  47. package/dist/src/adapters/pi-tools.d.ts +10 -0
  48. package/dist/src/adapters/pi-tools.js +72 -0
  49. package/dist/src/adapters/pi-tools.js.map +1 -0
  50. package/dist/src/adapters/process-agent-adapter.d.ts +24 -0
  51. package/dist/src/adapters/process-agent-adapter.js +111 -0
  52. package/dist/src/adapters/process-agent-adapter.js.map +1 -0
  53. package/dist/src/adapters/resolve-claude.d.ts +13 -0
  54. package/dist/src/adapters/resolve-claude.js +57 -0
  55. package/dist/src/adapters/resolve-claude.js.map +1 -0
  56. package/dist/src/adapters/resolve-codex.d.ts +13 -0
  57. package/dist/src/adapters/resolve-codex.js +48 -0
  58. package/dist/src/adapters/resolve-codex.js.map +1 -0
  59. package/dist/src/adapters/resolve-gemini.d.ts +13 -0
  60. package/dist/src/adapters/resolve-gemini.js +47 -0
  61. package/dist/src/adapters/resolve-gemini.js.map +1 -0
  62. package/dist/src/adapters/resolve-opencode.d.ts +11 -0
  63. package/dist/src/adapters/resolve-opencode.js +49 -0
  64. package/dist/src/adapters/resolve-opencode.js.map +1 -0
  65. package/dist/src/adapters/resolve-pi.d.ts +13 -0
  66. package/dist/src/adapters/resolve-pi.js +46 -0
  67. package/dist/src/adapters/resolve-pi.js.map +1 -0
  68. package/dist/src/adapters/run-options.d.ts +22 -0
  69. package/dist/src/adapters/run-options.js +48 -0
  70. package/dist/src/adapters/run-options.js.map +1 -0
  71. package/dist/src/adapters/spawn.d.ts +20 -0
  72. package/dist/src/adapters/spawn.js +16 -0
  73. package/dist/src/adapters/spawn.js.map +1 -0
  74. package/dist/src/agents/agent-manager.d.ts +98 -0
  75. package/dist/src/agents/agent-manager.js +433 -0
  76. package/dist/src/agents/agent-manager.js.map +1 -0
  77. package/dist/src/agents/attachments.d.ts +28 -0
  78. package/dist/src/agents/attachments.js +121 -0
  79. package/dist/src/agents/attachments.js.map +1 -0
  80. package/dist/src/bridge-context.d.ts +45 -0
  81. package/dist/src/bridge-context.js +2 -0
  82. package/dist/src/bridge-context.js.map +1 -0
  83. package/dist/src/bridge-status.d.ts +12 -0
  84. package/dist/src/bridge-status.js +17 -0
  85. package/dist/src/bridge-status.js.map +1 -0
  86. package/dist/src/bridge.d.ts +37 -0
  87. package/dist/src/bridge.js +446 -0
  88. package/dist/src/bridge.js.map +1 -0
  89. package/dist/src/cli.d.ts +2 -0
  90. package/dist/src/cli.js +194 -0
  91. package/dist/src/cli.js.map +1 -0
  92. package/dist/src/conversation/session-history.d.ts +27 -0
  93. package/dist/src/conversation/session-history.js +1082 -0
  94. package/dist/src/conversation/session-history.js.map +1 -0
  95. package/dist/src/conversation/thread-store.d.ts +74 -0
  96. package/dist/src/conversation/thread-store.js +366 -0
  97. package/dist/src/conversation/thread-store.js.map +1 -0
  98. package/dist/src/daemon-config.d.ts +123 -0
  99. package/dist/src/daemon-config.js +64 -0
  100. package/dist/src/daemon-config.js.map +1 -0
  101. package/dist/src/daemon-state.d.ts +27 -0
  102. package/dist/src/daemon-state.js +76 -0
  103. package/dist/src/daemon-state.js.map +1 -0
  104. package/dist/src/git/git-runner.d.ts +24 -0
  105. package/dist/src/git/git-runner.js +63 -0
  106. package/dist/src/git/git-runner.js.map +1 -0
  107. package/dist/src/git/git-service.d.ts +76 -0
  108. package/dist/src/git/git-service.js +435 -0
  109. package/dist/src/git/git-service.js.map +1 -0
  110. package/dist/src/handler-router.d.ts +34 -0
  111. package/dist/src/handler-router.js +67 -0
  112. package/dist/src/handler-router.js.map +1 -0
  113. package/dist/src/handlers/account-handler.d.ts +4 -0
  114. package/dist/src/handlers/account-handler.js +27 -0
  115. package/dist/src/handlers/account-handler.js.map +1 -0
  116. package/dist/src/handlers/agent-handler.d.ts +2 -0
  117. package/dist/src/handlers/agent-handler.js +8 -0
  118. package/dist/src/handlers/agent-handler.js.map +1 -0
  119. package/dist/src/handlers/bridge-control-handler.d.ts +2 -0
  120. package/dist/src/handlers/bridge-control-handler.js +64 -0
  121. package/dist/src/handlers/bridge-control-handler.js.map +1 -0
  122. package/dist/src/handlers/desktop-handler.d.ts +12 -0
  123. package/dist/src/handlers/desktop-handler.js +5 -0
  124. package/dist/src/handlers/desktop-handler.js.map +1 -0
  125. package/dist/src/handlers/git-handler.d.ts +2 -0
  126. package/dist/src/handlers/git-handler.js +82 -0
  127. package/dist/src/handlers/git-handler.js.map +1 -0
  128. package/dist/src/handlers/index.d.ts +8 -0
  129. package/dist/src/handlers/index.js +22 -0
  130. package/dist/src/handlers/index.js.map +1 -0
  131. package/dist/src/handlers/not-implemented.d.ts +10 -0
  132. package/dist/src/handlers/not-implemented.js +21 -0
  133. package/dist/src/handlers/not-implemented.js.map +1 -0
  134. package/dist/src/handlers/notifications-handler.d.ts +2 -0
  135. package/dist/src/handlers/notifications-handler.js +62 -0
  136. package/dist/src/handlers/notifications-handler.js.map +1 -0
  137. package/dist/src/handlers/params.d.ts +11 -0
  138. package/dist/src/handlers/params.js +72 -0
  139. package/dist/src/handlers/params.js.map +1 -0
  140. package/dist/src/handlers/project-handler.d.ts +2 -0
  141. package/dist/src/handlers/project-handler.js +6 -0
  142. package/dist/src/handlers/project-handler.js.map +1 -0
  143. package/dist/src/handlers/thread-context-handler.d.ts +2 -0
  144. package/dist/src/handlers/thread-context-handler.js +211 -0
  145. package/dist/src/handlers/thread-context-handler.js.map +1 -0
  146. package/dist/src/handlers/workspace-handler.d.ts +2 -0
  147. package/dist/src/handlers/workspace-handler.js +101 -0
  148. package/dist/src/handlers/workspace-handler.js.map +1 -0
  149. package/dist/src/hooks/claude-approval-hook.d.ts +7 -0
  150. package/dist/src/hooks/claude-approval-hook.js +95 -0
  151. package/dist/src/hooks/claude-approval-hook.js.map +1 -0
  152. package/dist/src/hooks/gemini-approval-hook.d.ts +7 -0
  153. package/dist/src/hooks/gemini-approval-hook.js +113 -0
  154. package/dist/src/hooks/gemini-approval-hook.js.map +1 -0
  155. package/dist/src/index.d.ts +62 -0
  156. package/dist/src/index.js +65 -0
  157. package/dist/src/index.js.map +1 -0
  158. package/dist/src/keyring-secret-store.d.ts +36 -0
  159. package/dist/src/keyring-secret-store.js +70 -0
  160. package/dist/src/keyring-secret-store.js.map +1 -0
  161. package/dist/src/lock-file.d.ts +18 -0
  162. package/dist/src/lock-file.js +60 -0
  163. package/dist/src/lock-file.js.map +1 -0
  164. package/dist/src/logger.d.ts +28 -0
  165. package/dist/src/logger.js +99 -0
  166. package/dist/src/logger.js.map +1 -0
  167. package/dist/src/pairing/pairing-code-service.d.ts +45 -0
  168. package/dist/src/pairing/pairing-code-service.js +183 -0
  169. package/dist/src/pairing/pairing-code-service.js.map +1 -0
  170. package/dist/src/projects/project-registry.d.ts +14 -0
  171. package/dist/src/projects/project-registry.js +60 -0
  172. package/dist/src/projects/project-registry.js.map +1 -0
  173. package/dist/src/push/push-sender.d.ts +21 -0
  174. package/dist/src/push/push-sender.js +96 -0
  175. package/dist/src/push/push-sender.js.map +1 -0
  176. package/dist/src/push/push-service.d.ts +122 -0
  177. package/dist/src/push/push-service.js +260 -0
  178. package/dist/src/push/push-service.js.map +1 -0
  179. package/dist/src/qr.d.ts +17 -0
  180. package/dist/src/qr.js +31 -0
  181. package/dist/src/qr.js.map +1 -0
  182. package/dist/src/secret-store.d.ts +23 -0
  183. package/dist/src/secret-store.js +27 -0
  184. package/dist/src/secret-store.js.map +1 -0
  185. package/dist/src/secure-device-state.d.ts +16 -0
  186. package/dist/src/secure-device-state.js +63 -0
  187. package/dist/src/secure-device-state.js.map +1 -0
  188. package/dist/src/service-installer.d.ts +57 -0
  189. package/dist/src/service-installer.js +254 -0
  190. package/dist/src/service-installer.js.map +1 -0
  191. package/dist/src/session-state.d.ts +14 -0
  192. package/dist/src/session-state.js +19 -0
  193. package/dist/src/session-state.js.map +1 -0
  194. package/dist/src/transport/crypto.d.ts +43 -0
  195. package/dist/src/transport/crypto.js +78 -0
  196. package/dist/src/transport/crypto.js.map +1 -0
  197. package/dist/src/transport/lan-server.d.ts +33 -0
  198. package/dist/src/transport/lan-server.js +105 -0
  199. package/dist/src/transport/lan-server.js.map +1 -0
  200. package/dist/src/transport/local-hosts.d.ts +17 -0
  201. package/dist/src/transport/local-hosts.js +30 -0
  202. package/dist/src/transport/local-hosts.js.map +1 -0
  203. package/dist/src/transport/mdns-advertiser.d.ts +83 -0
  204. package/dist/src/transport/mdns-advertiser.js +282 -0
  205. package/dist/src/transport/mdns-advertiser.js.map +1 -0
  206. package/dist/src/transport/message-io.d.ts +33 -0
  207. package/dist/src/transport/message-io.js +87 -0
  208. package/dist/src/transport/message-io.js.map +1 -0
  209. package/dist/src/transport/outbound-log.d.ts +24 -0
  210. package/dist/src/transport/outbound-log.js +78 -0
  211. package/dist/src/transport/outbound-log.js.map +1 -0
  212. package/dist/src/transport/relay-client.d.ts +19 -0
  213. package/dist/src/transport/relay-client.js +27 -0
  214. package/dist/src/transport/relay-client.js.map +1 -0
  215. package/dist/src/transport/secure-channel.d.ts +33 -0
  216. package/dist/src/transport/secure-channel.js +81 -0
  217. package/dist/src/transport/secure-channel.js.map +1 -0
  218. package/dist/src/transport/server-handshake.d.ts +49 -0
  219. package/dist/src/transport/server-handshake.js +137 -0
  220. package/dist/src/transport/server-handshake.js.map +1 -0
  221. package/dist/src/transport/session-handler.d.ts +19 -0
  222. package/dist/src/transport/session-handler.js +134 -0
  223. package/dist/src/transport/session-handler.js.map +1 -0
  224. package/dist/src/transport/session-registry.d.ts +58 -0
  225. package/dist/src/transport/session-registry.js +91 -0
  226. package/dist/src/transport/session-registry.js.map +1 -0
  227. package/dist/src/transport/trust-store.d.ts +23 -0
  228. package/dist/src/transport/trust-store.js +33 -0
  229. package/dist/src/transport/trust-store.js.map +1 -0
  230. package/dist/src/transport/ws-adapter.d.ts +7 -0
  231. package/dist/src/transport/ws-adapter.js +16 -0
  232. package/dist/src/transport/ws-adapter.js.map +1 -0
  233. package/dist/src/version.d.ts +1 -0
  234. package/dist/src/version.js +13 -0
  235. package/dist/src/version.js.map +1 -0
  236. package/dist/src/workspace/browse-service.d.ts +10 -0
  237. package/dist/src/workspace/browse-service.js +97 -0
  238. package/dist/src/workspace/browse-service.js.map +1 -0
  239. package/dist/src/workspace/checkpoint-service.d.ts +21 -0
  240. package/dist/src/workspace/checkpoint-service.js +219 -0
  241. package/dist/src/workspace/checkpoint-service.js.map +1 -0
  242. package/dist/src/workspace/path-guard.d.ts +7 -0
  243. package/dist/src/workspace/path-guard.js +51 -0
  244. package/dist/src/workspace/path-guard.js.map +1 -0
  245. package/dist/src/workspace/workspace-service.d.ts +8 -0
  246. package/dist/src/workspace/workspace-service.js +111 -0
  247. package/dist/src/workspace/workspace-service.js.map +1 -0
  248. package/package.json +46 -0
  249. package/scripts/extract-gemini-hook.mjs +16 -0
  250. package/scripts/fake-approval-bridge.mjs +23 -0
  251. package/scripts/install-service-linux.sh +38 -0
  252. package/scripts/install-service-macos.sh +38 -0
  253. package/scripts/install-service-windows.ps1 +26 -0
  254. package/scripts/test-gemini-hook-e2e.mjs +168 -0
  255. package/scripts/write-gemini-settings.mjs +31 -0
@@ -0,0 +1,168 @@
1
+ /**
2
+ * End-to-end smoke test for the Gemini approval hook.
3
+ *
4
+ * Spins up a tiny HTTP server that emulates the bridge's
5
+ * `POST /agent-hook/approval` endpoint, asks the bridge's
6
+ * `writeGeminiApprovalHook` to materialize the script, then runs the script
7
+ * with the exact JSON payload Gemini's `BeforeTool` event sends on stdin.
8
+ *
9
+ * Verifies:
10
+ * 1. Allow path → script exits 0, no stdout (Gemini's "allow" shape).
11
+ * 2. Deny path → script exits 0, stdout is `{ decision: "deny", reason }`.
12
+ * 3. No URL → script exits 0 with no stdout (defensive fail-open).
13
+ */
14
+ import { writeGeminiApprovalHook } from '../dist/src/hooks/gemini-approval-hook.js';
15
+ import { spawn } from 'node:child_process';
16
+ import { createServer } from 'node:http';
17
+ import { mkdtempSync, readFileSync } from 'node:fs';
18
+ import { tmpdir } from 'node:os';
19
+ import { join } from 'node:path';
20
+
21
+ let failures = 0;
22
+
23
+ function fail(msg) {
24
+ console.error(`✖ ${msg}`);
25
+ failures++;
26
+ }
27
+
28
+ function ok(msg) {
29
+ console.log(`✔ ${msg}`);
30
+ }
31
+
32
+ function runHook(scriptPath, env, stdin, timeoutMs = 5000) {
33
+ return new Promise((resolve) => {
34
+ const child = spawn('node', [scriptPath], {
35
+ env: { ...process.env, ...env },
36
+ stdio: ['pipe', 'pipe', 'pipe'],
37
+ windowsHide: true,
38
+ shell: false,
39
+ });
40
+ let stdout = '';
41
+ let stderr = '';
42
+ let settled = false;
43
+ const finish = (code) => {
44
+ if (settled) return;
45
+ settled = true;
46
+ resolve({ stdout, stderr, code });
47
+ };
48
+ child.stdout.on('data', (c) => (stdout += c.toString('utf-8')));
49
+ child.stderr.on('data', (c) => (stderr += c.toString('utf-8')));
50
+ child.on('close', finish);
51
+ child.on('error', (err) => finish(-1));
52
+ setTimeout(() => {
53
+ try {
54
+ child.kill();
55
+ } catch {
56
+ /* ignore */
57
+ }
58
+ }, timeoutMs);
59
+ child.stdin.write(stdin);
60
+ child.stdin.end();
61
+ });
62
+ }
63
+
64
+ const toolInput = JSON.stringify({
65
+ tool_name: 'write_file',
66
+ tool_input: { file_path: 'a.txt', content: 'x' },
67
+ session_id: 'sess-1',
68
+ cwd: '/tmp',
69
+ hook_event_name: 'BeforeTool',
70
+ });
71
+
72
+ async function main() {
73
+ const dir = mkdtempSync(join(tmpdir(), 'gemini-e2e-'));
74
+ const scriptPath = await writeGeminiApprovalHook(join(dir, 'hook.cjs'));
75
+ ok(`wrote hook script to ${scriptPath}`);
76
+ // Sanity: the writer output equals the embedded constant.
77
+ const written = readFileSync(scriptPath, 'utf-8');
78
+ if (written.length < 1000) fail(`hook script suspiciously short: ${written.length}b`);
79
+ else ok(`hook script is ${written.length}b`);
80
+
81
+ // Allow path.
82
+ const allowServer = createServer((_req, res) => {
83
+ res.writeHead(200, { 'content-type': 'application/json' });
84
+ res.end(JSON.stringify({ decision: 'allow' }));
85
+ });
86
+ await new Promise((r) => allowServer.listen(0, '127.0.0.1', r));
87
+ const allowAddr = allowServer.address();
88
+ const allowUrl = `http://127.0.0.1:${allowAddr.port}/agent-hook/approval`;
89
+ {
90
+ const r = await runHook(
91
+ scriptPath,
92
+ { UXNAN_HOOK_URL: allowUrl, UXNAN_HOOK_TOKEN: 't', UXNAN_HOOK_THREAD_ID: 'th' },
93
+ toolInput,
94
+ );
95
+ if (r.code === 0 && r.stdout === '') ok('allow → exit 0, empty stdout');
96
+ else fail(`allow path wrong: code=${r.code} stdout=${JSON.stringify(r.stdout)}`);
97
+ }
98
+ allowServer.close();
99
+
100
+ // Deny path.
101
+ const denyServer = createServer((_req, res) => {
102
+ res.writeHead(200, { 'content-type': 'application/json' });
103
+ res.end(JSON.stringify({ decision: 'deny', reason: 'blocked by phone' }));
104
+ });
105
+ await new Promise((r) => denyServer.listen(0, '127.0.0.1', r));
106
+ const denyAddr = denyServer.address();
107
+ const denyUrl = `http://127.0.0.1:${denyAddr.port}/agent-hook/approval`;
108
+ {
109
+ const r = await runHook(
110
+ scriptPath,
111
+ { UXNAN_HOOK_URL: denyUrl, UXNAN_HOOK_TOKEN: 't', UXNAN_HOOK_THREAD_ID: 'th' },
112
+ toolInput,
113
+ );
114
+ if (r.code !== 0) fail(`deny path exited non-zero: ${r.code}`);
115
+ try {
116
+ const j = JSON.parse(r.stdout);
117
+ if (j.decision === 'deny' && j.reason === 'blocked by phone') {
118
+ ok('deny → exit 0 + { decision: "deny", reason: … }');
119
+ } else fail(`deny payload wrong: ${JSON.stringify(j)}`);
120
+ } catch (e) {
121
+ fail(`deny stdout not JSON: ${r.stdout}`);
122
+ }
123
+ }
124
+ denyServer.close();
125
+
126
+ // No URL → defensive allow.
127
+ {
128
+ const r = await runHook(
129
+ scriptPath,
130
+ { UXNAN_HOOK_TOKEN: 't', UXNAN_HOOK_THREAD_ID: 'th' },
131
+ toolInput,
132
+ );
133
+ if (r.code === 0 && r.stdout === '') ok('no URL → exit 0, empty stdout (fail-open)');
134
+ else fail(`no-URL path wrong: code=${r.code} stdout=${JSON.stringify(r.stdout)}`);
135
+ }
136
+
137
+ // Bridge unreachable → fail-closed deny.
138
+ {
139
+ const r = await runHook(
140
+ scriptPath,
141
+ {
142
+ UXNAN_HOOK_URL: 'http://127.0.0.1:1/agent-hook/approval',
143
+ UXNAN_HOOK_TOKEN: 't',
144
+ UXNAN_HOOK_THREAD_ID: 'th',
145
+ },
146
+ toolInput,
147
+ );
148
+ if (r.code === 0) {
149
+ try {
150
+ const j = JSON.parse(r.stdout);
151
+ if (j.decision === 'deny') ok('unreachable → deny (fail-closed)');
152
+ else fail(`unreachable wrong payload: ${JSON.stringify(j)}`);
153
+ } catch (e) {
154
+ fail(`unreachable stdout not JSON: ${r.stdout}`);
155
+ }
156
+ } else fail(`unreachable wrong exit: ${r.code}`);
157
+ }
158
+ }
159
+
160
+ main().then(() => {
161
+ if (failures === 0) {
162
+ console.log('\nAll end-to-end checks passed.');
163
+ process.exit(0);
164
+ } else {
165
+ console.error(`\n${failures} check(s) failed.`);
166
+ process.exit(1);
167
+ }
168
+ });
@@ -0,0 +1,31 @@
1
+ import { mkdirSync, readFileSync, writeFileSync } from 'node:fs';
2
+ import { join } from 'node:path';
3
+
4
+ const cwd = process.argv[2];
5
+ const hookPath = process.argv[3];
6
+ if (!cwd || !hookPath) {
7
+ console.error('usage: write-gemini-settings.mjs <cwd> <hook-path>');
8
+ process.exit(2);
9
+ }
10
+
11
+ const settings = {
12
+ hooks: {
13
+ BeforeTool: [
14
+ {
15
+ matcher: '.*',
16
+ hooks: [
17
+ {
18
+ type: 'command',
19
+ name: 'uxnan-approval',
20
+ command: `node "${hookPath}"`,
21
+ },
22
+ ],
23
+ },
24
+ ],
25
+ },
26
+ };
27
+
28
+ mkdirSync(join(cwd, '.gemini'), { recursive: true });
29
+ const out = join(cwd, '.gemini', 'settings.json');
30
+ writeFileSync(out, JSON.stringify(settings, null, 2), 'utf-8');
31
+ console.log(out);