@rubytech/create-realagent 1.0.622 → 1.0.624

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 (26) hide show
  1. package/package.json +1 -1
  2. package/payload/platform/plugins/admin/mcp/dist/index.js +1 -1
  3. package/payload/platform/plugins/admin/mcp/dist/index.js.map +1 -1
  4. package/payload/platform/plugins/admin/skills/onboarding/SKILL.md +9 -12
  5. package/payload/platform/plugins/cloudflare/PLUGIN.md +35 -22
  6. package/payload/platform/plugins/cloudflare/mcp/dist/index.js +13 -801
  7. package/payload/platform/plugins/cloudflare/mcp/dist/index.js.map +1 -1
  8. package/payload/platform/plugins/cloudflare/mcp/dist/lib/cloudflared.d.ts.map +1 -1
  9. package/payload/platform/plugins/cloudflare/mcp/dist/lib/cloudflared.js +1 -0
  10. package/payload/platform/plugins/cloudflare/mcp/dist/lib/cloudflared.js.map +1 -1
  11. package/payload/platform/plugins/cloudflare/mcp/dist/lib/setup-orchestrator.d.ts +90 -0
  12. package/payload/platform/plugins/cloudflare/mcp/dist/lib/setup-orchestrator.d.ts.map +1 -0
  13. package/payload/platform/plugins/cloudflare/mcp/dist/lib/setup-orchestrator.js +550 -0
  14. package/payload/platform/plugins/cloudflare/mcp/dist/lib/setup-orchestrator.js.map +1 -0
  15. package/payload/platform/plugins/cloudflare/references/dashboard-guide.md +108 -0
  16. package/payload/platform/plugins/cloudflare/references/manual-setup.md +445 -0
  17. package/payload/platform/plugins/cloudflare/references/reset-guide.md +118 -0
  18. package/payload/platform/plugins/cloudflare/scripts/reset-tunnel.sh +65 -0
  19. package/payload/platform/plugins/cloudflare/scripts/setup-tunnel.sh +244 -0
  20. package/payload/platform/plugins/cloudflare/skills/setup-tunnel/SKILL.md +81 -41
  21. package/payload/platform/plugins/docs/references/cloudflare.md +93 -29
  22. package/payload/platform/templates/agents/admin/IDENTITY.md +14 -0
  23. package/payload/platform/templates/specialists/agents/personal-assistant.md +9 -9
  24. package/payload/server/server.js +6 -11
  25. package/payload/platform/config/cloudflared.yml +0 -17
  26. package/payload/platform/plugins/cloudflare/references/setup-guide.md +0 -132
@@ -0,0 +1,550 @@
1
+ /**
2
+ * Cloudflare tunnel setup — deterministic orchestrator.
3
+ *
4
+ * Task 547: the tunnel setup flow (login → create → enable → verify) has a
5
+ * deterministic shape. Every branch is driven by a structured signal —
6
+ * either `tunnel-status`'s `unhealthyReason` enum, or the operator's
7
+ * literal input (domain name, subdomain names, "done"). Before Task 547
8
+ * the flow was driven by the LLM reading SKILL.md prose and picking the
9
+ * next `tunnel-*` tool; that allowed a dozen+ recurrences where the LLM
10
+ * forked, dispatched Playwright to the Cloudflare dashboard, or fabricated
11
+ * a `cp cert.pem` workaround. None of those deviations are reachable here
12
+ * because the LLM no longer chooses — this module does.
13
+ *
14
+ * Contract:
15
+ * - `runOrchestrator(input)` is the single entry point. One call advances
16
+ * one deterministic step. The LLM's only job is to pass the operator's
17
+ * literal input through (domain, "done", etc.) and relay the structured
18
+ * result verbatim.
19
+ * - Every transition emits a `[cloudflare:setup-run:phase-transition]` log
20
+ * line. Every internal sub-tool call emits
21
+ * `[cloudflare:setup-run:subtool-call]`. Every terminal state emits
22
+ * `[cloudflare:setup-run:terminal]`. These give a single grep path
23
+ * (`[cloudflare:setup-run:`) to the full transition chain post-hoc.
24
+ * - State is persisted at `~/{configDir}/cloudflared/setup.state.json`
25
+ * alongside `tunnel.state` and `login.state`. On corruption or absence
26
+ * the orchestrator reconciles from live `tunnel-status` + operator
27
+ * input (no "state file not found" crash).
28
+ * - The tool-surface gate in `platform/ui/app/lib/tool-surface-filter.ts`
29
+ * reads this state file every turn and removes the raw `tunnel-*`,
30
+ * Playwright, Bash, Write, Edit tools from the LLM's menu whenever
31
+ * `phase !== 'idle' && phase !== 'healthy'`. No LLM regression can
32
+ * re-enter the loop because the tools that enabled it are literally
33
+ * absent from the menu.
34
+ */
35
+ import { existsSync, readFileSync, writeFileSync, mkdirSync, unlinkSync, } from "node:fs";
36
+ import { join } from "node:path";
37
+ import { homedir } from "node:os";
38
+ import * as cloudflared from "./cloudflared.js";
39
+ import { deviceUrlBlock } from "../../../../../lib/device-url/dist/index.js";
40
+ import { hostname as osHostname } from "node:os";
41
+ function initialState() {
42
+ return {
43
+ phase: "idle",
44
+ domain: null,
45
+ adminSubdomain: null,
46
+ publicSubdomain: null,
47
+ tunnelId: null,
48
+ phaseEnteredAt: new Date().toISOString(),
49
+ unhealthyReason: null,
50
+ };
51
+ }
52
+ // ---------------------------------------------------------------------------
53
+ // State file I/O
54
+ // ---------------------------------------------------------------------------
55
+ /**
56
+ * The orchestrator state file path. Consumers OUTSIDE this module (e.g. the
57
+ * tool-surface filter in `platform/ui/app/lib/tool-surface-filter.ts`) also
58
+ * need this path to read the phase. Exported so the two sides cannot drift.
59
+ *
60
+ * Brand-aware via loadBrand(); falls back to `.maxy` when PLATFORM_ROOT is
61
+ * unset. The fallback matters only for the tool-surface filter running
62
+ * outside the cloudflare MCP server's process — the MCP server always has
63
+ * PLATFORM_ROOT set by the claude-agent spawn.
64
+ */
65
+ export function setupStatePath(configDirOverride) {
66
+ const configDir = configDirOverride ?? (() => {
67
+ try {
68
+ return cloudflared.loadBrand().configDir;
69
+ }
70
+ catch {
71
+ return ".maxy";
72
+ }
73
+ })();
74
+ return join(homedir(), configDir, "cloudflared", "setup.state.json");
75
+ }
76
+ function readState() {
77
+ const path = setupStatePath();
78
+ if (!existsSync(path))
79
+ return initialState();
80
+ try {
81
+ const parsed = JSON.parse(readFileSync(path, "utf-8"));
82
+ // Loose validation — if the file is malformed, treat as idle and rely
83
+ // on `tunnel-status` reconciliation to map back to the right phase.
84
+ // A "state file not found" crash would be a silent failure mode the
85
+ // operator cannot recover from; mapping to idle is loud-failure (log)
86
+ // but non-fatal.
87
+ if (typeof parsed?.phase !== "string") {
88
+ console.error(`[cloudflare:setup-run:state-malformed] path=${path} — resetting to idle`);
89
+ return initialState();
90
+ }
91
+ return {
92
+ phase: parsed.phase,
93
+ domain: parsed.domain ?? null,
94
+ adminSubdomain: parsed.adminSubdomain ?? null,
95
+ publicSubdomain: parsed.publicSubdomain ?? null,
96
+ tunnelId: parsed.tunnelId ?? null,
97
+ phaseEnteredAt: parsed.phaseEnteredAt ?? new Date().toISOString(),
98
+ unhealthyReason: parsed.unhealthyReason ?? null,
99
+ };
100
+ }
101
+ catch (err) {
102
+ console.error(`[cloudflare:setup-run:state-read-error] path=${path} err=${err instanceof Error ? err.message : String(err)}`);
103
+ return initialState();
104
+ }
105
+ }
106
+ function writeState(state) {
107
+ const path = setupStatePath();
108
+ mkdirSync(join(homedir(), cloudflared.loadBrand().configDir, "cloudflared"), { recursive: true });
109
+ writeFileSync(path, JSON.stringify(state, null, 2), { mode: 0o600 });
110
+ }
111
+ function deleteState() {
112
+ const path = setupStatePath();
113
+ try {
114
+ unlinkSync(path);
115
+ }
116
+ catch (err) {
117
+ const code = err.code;
118
+ if (code !== "ENOENT") {
119
+ console.error(`[cloudflare:setup-run:state-delete-error] path=${path} err=${err}`);
120
+ }
121
+ }
122
+ }
123
+ function transition(state, next, reason) {
124
+ if (state.phase !== next) {
125
+ console.error(`[cloudflare:setup-run:phase-transition] from=${state.phase} to=${next} reason=${reason}`);
126
+ }
127
+ const updated = {
128
+ ...state,
129
+ phase: next,
130
+ phaseEnteredAt: new Date().toISOString(),
131
+ };
132
+ writeState(updated);
133
+ return updated;
134
+ }
135
+ // ---------------------------------------------------------------------------
136
+ // Read-only status
137
+ //
138
+ // Exposed so the tool-surface filter (outside this MCP server) can read
139
+ // orchestrator phase without advancing it, and so the companion
140
+ // `cloudflare-setup-status` tool can surface it to the admin agent.
141
+ // ---------------------------------------------------------------------------
142
+ export function readOrchestratorState() {
143
+ return readState();
144
+ }
145
+ /**
146
+ * Whether the tool-surface filter should treat this device as "setup in
147
+ * progress". Matches the gate condition named in the task file: the filter
148
+ * lifts whenever phase is idle or terminal (healthy / unhealthy).
149
+ */
150
+ export function isSetupActive(state) {
151
+ return state.phase !== "idle" && state.phase !== "healthy" && state.phase !== "unhealthy";
152
+ }
153
+ function subtoolLog(name, phase) {
154
+ console.error(`[cloudflare:setup-run:subtool-call] subtool=${name} phase=${phase}`);
155
+ }
156
+ function terminalLog(outcome, reason) {
157
+ console.error(`[cloudflare:setup-run:terminal] outcome=${outcome} reason=${reason}`);
158
+ }
159
+ function operatorInputLog(phase, repr) {
160
+ console.error(`[cloudflare:setup-run:operator-input] phase=${phase} input=${repr}`);
161
+ }
162
+ // ---------------------------------------------------------------------------
163
+ // Subdomain validation
164
+ //
165
+ // The operator types subdomain labels directly. Reject anything that would
166
+ // break `cloudflared tunnel route dns` before shelling out. Labels: letters,
167
+ // digits, hyphens; 1-63 chars; no leading/trailing hyphen.
168
+ // ---------------------------------------------------------------------------
169
+ const SUBDOMAIN_PATTERN = /^[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?$/i;
170
+ function validateSubdomain(label, name) {
171
+ if (label.length === 0)
172
+ return `${name} is empty`;
173
+ if (label.includes("."))
174
+ return `${name} must be a single label (got "${label}"). Use just the prefix, e.g. "admin" — not "admin.example.com".`;
175
+ if (!SUBDOMAIN_PATTERN.test(label))
176
+ return `${name} "${label}" has invalid characters. Use letters, digits, and hyphens only.`;
177
+ return null;
178
+ }
179
+ // ---------------------------------------------------------------------------
180
+ // Card builder — the `maxy-device-url` block wrapped in intent prose
181
+ // ---------------------------------------------------------------------------
182
+ function signInCard(authUrl, accountHint) {
183
+ const block = deviceUrlBlock({
184
+ url: authUrl,
185
+ intent: "Sign in to Cloudflare",
186
+ hostname: osHostname(),
187
+ });
188
+ return (`Cloudflare sign-in started on this device.\n\n` +
189
+ `${block}\n\n` +
190
+ `Click the button to open the sign-in page on this device's browser. ` +
191
+ `Pick the account that owns ${accountHint} — the account name is in the ` +
192
+ `top-left of the Cloudflare page, next to the little orange cloud — then ` +
193
+ `click Authorize. Tell me when you have finished.`);
194
+ }
195
+ function accountSwitchCard(hostname) {
196
+ return (`That Cloudflare account does not own ${hostname}.\n\n` +
197
+ `Open Cloudflare in your browser. Look at the name in the top-left. ` +
198
+ `If it is not the account that owns ${hostname}, switch to the correct ` +
199
+ `account using the top-left dropdown. Once the correct account is showing, ` +
200
+ `tell me and I will start the sign-in again.`);
201
+ }
202
+ // ---------------------------------------------------------------------------
203
+ // The orchestrator entry point
204
+ //
205
+ // Every call:
206
+ // (1) Reads persisted state.
207
+ // (2) Probes live tunnel-status.
208
+ // (3) Reconciles — if status shows we're further along than persisted,
209
+ // advance persisted to match.
210
+ // (4) Handles current phase.
211
+ //
212
+ // Step (3) is what makes re-entry and cold-start correct. An operator who
213
+ // deletes the state file mid-flow gets reconciled to the right phase on
214
+ // the next call because tunnel-status is the ground truth. Fresh-install
215
+ // with no state and no cert starts at `idle` → `awaiting-domain`. Fresh-
216
+ // install with no state but cert already bound (migration from a prior
217
+ // setup) maps to `awaiting-hostnames`.
218
+ // ---------------------------------------------------------------------------
219
+ export async function runOrchestrator(input) {
220
+ let state = readState();
221
+ console.error(`[cloudflare:setup-run:enter] phase=${state.phase} domain=${state.domain ?? "none"}`);
222
+ // Log operator input once, terse — helpful post-hoc when diagnosing how
223
+ // the state transitioned. Never log secrets; domain is public by design.
224
+ const inputRepr = JSON.stringify({
225
+ domain: input.domain ?? null,
226
+ adminSubdomain: input.adminSubdomain ?? null,
227
+ publicSubdomain: input.publicSubdomain ?? null,
228
+ confirmed: input.confirmed ?? false,
229
+ });
230
+ operatorInputLog(state.phase, inputRepr);
231
+ // ---------------------------------------------------------------------
232
+ // Reconcile persisted phase against live tunnel-status.
233
+ //
234
+ // The reconciliation rule: tunnel-status is always right. If status
235
+ // says we're healthy, we are healthy (regardless of persisted phase).
236
+ // If status says we have a cert but persisted says we still need one,
237
+ // we skip ahead to `awaiting-hostnames`. If persisted domain is null
238
+ // but input provides one, adopt it.
239
+ // ---------------------------------------------------------------------
240
+ if (input.domain) {
241
+ const domainTrimmed = input.domain.trim().replace(/^https?:\/\//, "").replace(/\/.*$/, "");
242
+ if (domainTrimmed.length === 0) {
243
+ return { text: "Invalid domain — please give me the bare domain, e.g. maxy.bot.", phase: state.phase, healthy: false };
244
+ }
245
+ if (state.domain !== domainTrimmed) {
246
+ state = { ...state, domain: domainTrimmed };
247
+ writeState(state);
248
+ }
249
+ }
250
+ if (!state.domain && state.phase === "idle") {
251
+ state = transition(state, "awaiting-domain", "initial-entry-no-domain");
252
+ return {
253
+ text: "What domain do you want to use for the tunnel? For example: maxy.bot.",
254
+ phase: state.phase,
255
+ healthy: false,
256
+ };
257
+ }
258
+ // Probe live tunnel-status. When `state.domain` is null we pass undefined
259
+ // and let cloudflared use persisted state.
260
+ subtoolLog("tunnel-status", state.phase);
261
+ const status = await cloudflared.getStatus(state.domain ?? undefined);
262
+ // Terminal healthy short-circuit — same answer from any phase.
263
+ if (status.healthy) {
264
+ // A second call after initial healthy convergence is a no-op. Log once
265
+ // as terminal so the review-detector can confirm the flow closed.
266
+ if (state.phase !== "healthy") {
267
+ state = transition(state, "healthy", "status-healthy");
268
+ terminalLog("healthy", "all-probes-ok");
269
+ }
270
+ else {
271
+ console.error(`[cloudflare:setup-run:no-op] reason=already-healthy`);
272
+ }
273
+ const urls = status.probes.map((p) => ` https://${p.hostname}`).join("\n");
274
+ return {
275
+ text: `Tunnel is healthy.\n\n${urls}`,
276
+ phase: state.phase,
277
+ healthy: true,
278
+ };
279
+ }
280
+ // --- From here: phase-specific advance logic -------------------------
281
+ // If we have a cert and it matches the binding, we are past the login
282
+ // phase. Don't re-spawn login.
283
+ const authedAndReady = status.hasCert && status.bound;
284
+ // Determine next action from (persisted phase, status, input).
285
+ switch (state.phase) {
286
+ case "awaiting-domain": {
287
+ // We have a domain now (input.domain was applied above). Advance.
288
+ if (authedAndReady) {
289
+ state = transition(state, "awaiting-hostnames", "cert-already-bound");
290
+ }
291
+ else {
292
+ state = transition(state, "awaiting-signin", "no-cert-spawning-login");
293
+ }
294
+ return continueFromPhase(state, input, status);
295
+ }
296
+ case "idle": {
297
+ // Cold start with domain already supplied (rare — the check above
298
+ // only transitions to awaiting-domain when there's no domain).
299
+ if (authedAndReady) {
300
+ state = transition(state, "awaiting-hostnames", "cert-already-bound");
301
+ }
302
+ else {
303
+ state = transition(state, "awaiting-signin", "cold-start-needs-login");
304
+ }
305
+ return continueFromPhase(state, input, status);
306
+ }
307
+ case "awaiting-signin": {
308
+ if (authedAndReady) {
309
+ state = transition(state, "awaiting-hostnames", "sign-in-complete");
310
+ return continueFromPhase(state, input, status);
311
+ }
312
+ return continueFromPhase(state, input, status);
313
+ }
314
+ case "awaiting-account-switch": {
315
+ // Operator confirms they have switched accounts. Re-login with force.
316
+ if (input.confirmed) {
317
+ subtoolLog("tunnel-login-force", state.phase);
318
+ cloudflared.resetAuth();
319
+ const res = await cloudflared.tunnelLogin();
320
+ state = transition(state, "awaiting-signin", "account-switch-relogin");
321
+ return {
322
+ text: signInCard(res.authUrl, state.domain ?? "your domain"),
323
+ phase: state.phase,
324
+ healthy: false,
325
+ };
326
+ }
327
+ return {
328
+ text: accountSwitchCard(state.domain ?? "your domain"),
329
+ phase: state.phase,
330
+ healthy: false,
331
+ };
332
+ }
333
+ case "awaiting-hostnames": {
334
+ return continueFromPhase(state, input, status);
335
+ }
336
+ case "enabling": {
337
+ // Re-entry after crash in the middle of create/enable. Probe told
338
+ // us we're not healthy; fall through to re-run enable.
339
+ return continueFromPhase(state, input, status);
340
+ }
341
+ case "healthy": {
342
+ // Unreachable — the healthy short-circuit above returns before here.
343
+ // Keep for exhaustiveness.
344
+ return { text: "Tunnel is healthy.", phase: state.phase, healthy: true };
345
+ }
346
+ case "unhealthy": {
347
+ // Re-invoking after an unhealthy terminal re-probes and either
348
+ // converges (returns healthy above) or re-emits the same reason.
349
+ const reason = status.unhealthyReason ?? "unknown";
350
+ return {
351
+ text: `Tunnel is not healthy. Reason: ${reason}. Re-running setup from the current state — ask me again in a minute.`,
352
+ phase: state.phase,
353
+ healthy: false,
354
+ };
355
+ }
356
+ }
357
+ }
358
+ /**
359
+ * Handle a phase when it is the natural next step (or a fall-through from
360
+ * a reconciliation). Kept separate from `runOrchestrator`'s switch so the
361
+ * reconciliation cases above can transition and then delegate here without
362
+ * nested switching.
363
+ */
364
+ async function continueFromPhase(state, input, status) {
365
+ switch (state.phase) {
366
+ case "awaiting-signin": {
367
+ subtoolLog("tunnel-login", state.phase);
368
+ const res = await cloudflared.tunnelLogin();
369
+ return {
370
+ text: signInCard(res.authUrl, state.domain ?? "your domain"),
371
+ phase: state.phase,
372
+ healthy: false,
373
+ };
374
+ }
375
+ case "awaiting-hostnames": {
376
+ // Need admin subdomain from the operator.
377
+ if (!input.adminSubdomain) {
378
+ return {
379
+ text: `What subdomain should the admin interface use? ` +
380
+ `For example "admin" → admin.${state.domain}. ` +
381
+ `Optionally, what subdomain for the public chat? ` +
382
+ `For example "public" → public.${state.domain}. ` +
383
+ `Leave blank to skip the public chat.`,
384
+ phase: state.phase,
385
+ healthy: false,
386
+ };
387
+ }
388
+ const adminErr = validateSubdomain(input.adminSubdomain, "admin subdomain");
389
+ if (adminErr)
390
+ return { text: adminErr, phase: state.phase, healthy: false };
391
+ let publicSubdomain = null;
392
+ if (input.publicSubdomain) {
393
+ const pubErr = validateSubdomain(input.publicSubdomain, "public subdomain");
394
+ if (pubErr)
395
+ return { text: pubErr, phase: state.phase, healthy: false };
396
+ publicSubdomain = input.publicSubdomain;
397
+ }
398
+ if (publicSubdomain && publicSubdomain === input.adminSubdomain) {
399
+ return {
400
+ text: `Admin and public subdomains must differ — both given as "${publicSubdomain}".`,
401
+ phase: state.phase,
402
+ healthy: false,
403
+ };
404
+ }
405
+ // Persist the choices so re-entry survives mid-call crashes.
406
+ state = { ...state, adminSubdomain: input.adminSubdomain, publicSubdomain };
407
+ writeState(state);
408
+ // Transition to enabling and run create + enable + verify.
409
+ state = transition(state, "enabling", "hostnames-received");
410
+ try {
411
+ // tunnel-create
412
+ subtoolLog("tunnel-create", state.phase);
413
+ const tunnelName = (state.domain ?? "tunnel").replace(/\./g, "-");
414
+ const tunnel = cloudflared.createTunnelCli(tunnelName);
415
+ const adminHostname = `${input.adminSubdomain}.${state.domain}`;
416
+ const publicHostname = publicSubdomain ? `${publicSubdomain}.${state.domain}` : null;
417
+ const hostnames = publicHostname ? [adminHostname, publicHostname] : [adminHostname];
418
+ const platformPort = parseInt(process.env.PLATFORM_PORT ?? "19200", 10);
419
+ const configPath = cloudflared.writeLocalConfig(tunnel.tunnelId, tunnel.credentialsPath, hostnames, platformPort);
420
+ cloudflared.saveTunnelIdentity({
421
+ tunnelId: tunnel.tunnelId,
422
+ tunnelName: tunnel.tunnelName,
423
+ domain: state.domain ?? "",
424
+ configPath,
425
+ credentialsPath: tunnel.credentialsPath,
426
+ adminHostname,
427
+ publicHostname,
428
+ });
429
+ state = { ...state, tunnelId: tunnel.tunnelId };
430
+ writeState(state);
431
+ // Route DNS for each hostname.
432
+ subtoolLog("tunnel-route-dns", state.phase);
433
+ for (const h of hostnames) {
434
+ await cloudflared.routeDnsCli(tunnel.tunnelId, h);
435
+ }
436
+ // tunnel-enable
437
+ subtoolLog("tunnel-enable", state.phase);
438
+ cloudflared.startTunnel({
439
+ tunnelId: tunnel.tunnelId,
440
+ tunnelName: tunnel.tunnelName,
441
+ domain: state.domain ?? "",
442
+ configPath,
443
+ credentialsPath: tunnel.credentialsPath,
444
+ });
445
+ // Poll `tunnel-status` until healthy or a hard reason surfaces.
446
+ // Same retry budget as the standalone tunnel-enable tool's verify
447
+ // loop: 6 attempts × 5s = 30s.
448
+ for (let attempt = 0; attempt < 6; attempt++) {
449
+ if (attempt > 0)
450
+ await new Promise((r) => setTimeout(r, 5000));
451
+ const probe = await cloudflared.getStatus(state.domain ?? undefined);
452
+ if (probe.healthy) {
453
+ state = transition(state, "healthy", "create-enable-verified");
454
+ terminalLog("healthy", "create-enable-verified");
455
+ const urls = probe.probes.map((p) => ` https://${p.hostname}`).join("\n");
456
+ return { text: `Tunnel is healthy.\n\n${urls}`, phase: state.phase, healthy: true };
457
+ }
458
+ if (probe.unhealthyReason === "bound-account-does-not-own-hostname") {
459
+ state = transition(state, "awaiting-account-switch", "post-enable-wrong-account");
460
+ return {
461
+ text: accountSwitchCard(state.domain ?? "your domain"),
462
+ phase: state.phase,
463
+ healthy: false,
464
+ };
465
+ }
466
+ }
467
+ // Couldn't verify within retry budget — emit a diagnostic unhealthy
468
+ // terminal. Operator can re-invoke cloudflare-setup-run later; if
469
+ // the edge caught up by then it will converge to healthy.
470
+ const final = await cloudflared.getStatus(state.domain ?? undefined);
471
+ const reason = final.unhealthyReason ?? "edge-not-yet-reachable";
472
+ state = { ...state, unhealthyReason: reason };
473
+ state = transition(state, "unhealthy", `verify-failed-${reason}`);
474
+ terminalLog("unhealthy", reason);
475
+ return {
476
+ text: `Tunnel created and started but the end-to-end probe has not converged yet ` +
477
+ `(reason: ${reason}). DNS propagation can take 1-5 minutes. Ask me to set up ` +
478
+ `the tunnel again in a minute to re-probe.`,
479
+ phase: state.phase,
480
+ healthy: false,
481
+ };
482
+ }
483
+ catch (err) {
484
+ // tunnel-create shelled refusal — detect the wrong-account class
485
+ // and route to awaiting-account-switch. Other errors bubble up as
486
+ // unhealthy with the message.
487
+ if (err instanceof cloudflared.CloudflareRefusalError) {
488
+ const reason = err.refusal.reason;
489
+ if (reason === "post-flight-fqdn-mismatch" ||
490
+ reason === "hostname-zone-not-routable" ||
491
+ reason === "account-drift" ||
492
+ reason === "unbound-device") {
493
+ state = transition(state, "awaiting-account-switch", `refusal-${reason}`);
494
+ return {
495
+ text: accountSwitchCard(state.domain ?? "your domain"),
496
+ phase: state.phase,
497
+ healthy: false,
498
+ };
499
+ }
500
+ }
501
+ const msg = err instanceof Error ? err.message : String(err);
502
+ state = { ...state, unhealthyReason: "create-or-enable-failed" };
503
+ state = transition(state, "unhealthy", "create-or-enable-threw");
504
+ terminalLog("unhealthy", "create-or-enable-threw");
505
+ return {
506
+ text: `Tunnel setup failed: ${msg}`,
507
+ phase: state.phase,
508
+ healthy: false,
509
+ };
510
+ }
511
+ }
512
+ case "enabling": {
513
+ // Re-enter mid-enable after a crash. Delegate back to
514
+ // awaiting-hostnames with the persisted subdomains as "input" so
515
+ // the create/enable/verify block runs again. Idempotent: tunnel
516
+ // create reuses an existing tunnel by name; DNS route is overwritten.
517
+ if (state.adminSubdomain) {
518
+ const recoveredInput = {
519
+ adminSubdomain: state.adminSubdomain,
520
+ publicSubdomain: state.publicSubdomain ?? undefined,
521
+ };
522
+ const recoveredState = transition(state, "awaiting-hostnames", "re-entry-after-enabling-crash");
523
+ return continueFromPhase(recoveredState, recoveredInput, status);
524
+ }
525
+ // Somehow hit enabling without a stored adminSubdomain — reset to
526
+ // awaiting-hostnames and ask again.
527
+ state = transition(state, "awaiting-hostnames", "re-entry-missing-subdomains");
528
+ return continueFromPhase(state, input, status);
529
+ }
530
+ case "idle":
531
+ case "awaiting-domain":
532
+ case "awaiting-account-switch":
533
+ case "healthy":
534
+ case "unhealthy":
535
+ // Unreachable in this helper — the top-level switch handles these.
536
+ return {
537
+ text: `Unexpected phase: ${state.phase}. Ask me again.`,
538
+ phase: state.phase,
539
+ healthy: false,
540
+ };
541
+ }
542
+ }
543
+ // ---------------------------------------------------------------------------
544
+ // Reset — exposed for tests and for the hypothetical "start over" admin flow
545
+ // ---------------------------------------------------------------------------
546
+ export function resetOrchestrator() {
547
+ console.error(`[cloudflare:setup-run:reset]`);
548
+ deleteState();
549
+ }
550
+ //# sourceMappingURL=setup-orchestrator.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"setup-orchestrator.js","sourceRoot":"","sources":["../../src/lib/setup-orchestrator.ts"],"names":[],"mappings":"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"}