- index.html title 'FlowMaster · Mission Control' + theme-color #1a2740 - Settings About box names canvas.flow-master.ai as the canonical URL - README rewritten: live URL up top, deploy section reflects merged PR + Cloudflare A record added (this session), end-to-end serving with cert-manager TLS + nginx /api/* proxy to demo.flow-master.ai - .dockerignore added (excludes node_modules, dist, qa/screenshots, .git, bak files) — keeps the build context lean - qa/smoke_canvas.mjs (new) — real-URL Playwright smoke against https://canvas.flow-master.ai with --host-resolver-rules so the local DNS cache can't shadow the freshly-published A record. 8/8 PASS, 0 console errors: ✓ landing renders at canvas.flow-master.ai ✓ hero title present ✓ scenario cards >= 7 (count=7) ✓ mission control loaded ✓ blueprint canvas mounted ✓ blueprint nodes rendered (nodes=3) ✓ LIVE mode resolves against canvas /api proxy ✓ at least one /api call observed (count=1) Deployment chain: shad/flowmaster-mission-control-demo @ dba1eb3 → docker buildx on build01 (linux/amd64 native) → gitea.flow-master.ai/shad/mission-control-demo:sha-dba1eb3 → FM06/flowmaster-ops PR #1164 (merged) → kubectl apply -n demo (ArgoCD was Degraded due to unrelated fm-shell env-var dupe; bypassed by applying directly) → 2/2 mc-mission-control pods Running → cluster ingress + cert-manager + Cloudflare A record → https://canvas.flow-master.ai serves the SPA + proxies /api to demo.flow-master.ai Confidence: high Scope-risk: narrow (demo namespace only; no shared baseline mutation) Not-tested: long-term cert renewal cycle (will happen automatically; cert-manager letsencrypt-prod-dns issuer is already proven by hakeem)
FlowMaster — Mission Control
Live at https://canvas.flow-master.ai · source on Gitea
The proper FlowMaster frontend: an operations cockpit for every process the company runs. Industrial-blueprint doctrine — paper canvas, navy frame, amber accent, 1px hairlines, square edges, monospace operational density.
What this is (and isn't)
Is:
- A single-page React 19 app (Vite + ReactFlow + cmdk + framer-motion + zustand
- dagre).
- Polished command-center for any FlowMaster process: graph, queue, inspector, tour, command palette, live-mode toggle, run history.
- Two data modes:
- SNAPSHOT (default): bundled
src/scenarios.json, captured fromdemo.flow-master.ai. Fast, offline, deterministic. - LIVE: in-browser fetch from the same backend via the API client at
src/lib/api.ts(dev-login → bearer →/api/ea2/work-items→/api/ea2/process-definitions/{k}/graph→/api/runtime/transactions/{id}). Loading and error states are wired throughsrc/scenes/MissionControl.tsx.
- SNAPSHOT (default): bundled
Isn't:
- Not a replacement for the existing demo at https://demo.flow-master.ai (Next.js fm-shell). This is a separate command-center experience.
- Not multi-tenant. No auth UI, no tenancy switcher, no settings.
- Not wired to actually mutate state. Every action button (start runtime, dispatch agent, approve, decline, confirm) is preview-only and fires a toast naming the endpoint that would make it real.
Scenarios in the catalog
| id | mode | source |
|---|---|---|
| procurement | live | Purchase Requisition → PO (pr_to_po_def on demo.flow-master.ai) |
| extra-1 | live | Atlas F1 Fresh (procurement variant) |
| extra-2 | live | Atlas F1 Fresh (procurement variant) |
| ar | blueprint | AR · Customer Refund Approval |
| hcm | blueprint | HCM · New Hire Onboarding |
| gl | blueprint | GL · Period-End Close |
| service | blueprint | Service Ops · Customer Incident |
Live = backed by a real EA2 process definition currently in the demo backend, with real runtime transactions and a real work-item queue.
Blueprint = hand-modelled in the same typed format. Identical UI surface;
the backend just doesn't have a runnable definition for it yet. Internal
metadata carries isSynthetic: true for provenance audits.
How honesty is enforced in this code
This pass came out of an Oracle review that called out theatrical bits in the prior version. Safeguards now in place:
- No invented numbers.
src/components/Telemetry.tsxderives every value from the active scenarios (SLA = 1 - errored / cases, agent acceptance = fraction of agent runs not inproposed). The throughput sparkline plots the actualrunningrollup over time, not a sine wave. The "ui tick" dot is labelled as a UI heartbeat and tooltip-named as such. - No fake buttons. Every preview-only action fires a toast that names the
endpoint that would make it real, and carries a
previewmarker. - No misleading tour copy. Blueprint tours frame the scenario as an industry blueprint, not "we don't have this yet".
- Mode is always visible. Topbar has a
SNAPSHOT/LIVEpill, last-fetch age, and a refresh button when live.
Live-mode mechanics (the CORS gotcha)
demo.flow-master.ai does not advertise CORS headers for arbitrary origins.
src/lib/api.ts therefore uses an empty baseUrl everywhere (both dev
and prod). All /api/* requests are same-origin from the browser's
perspective; whatever is serving the page is responsible for proxying them
to the backend.
- Dev (
pnpm dev):vite.config.tsproxies/api/*to${VITE_FM_BASE:-https://demo.flow-master.ai}. - Prod (Docker image): the bundled
nginx.confreverse-proxies/api/*tohttps://demo.flow-master.ai. The image is intended to sit behind thecanvas.flow-master.aiingress (seeFM06/flowmaster-opsoverlay). - Anywhere else: set
VITE_FM_BASE=https://your-backendat build time and accept that browsers will reject the cross-origin call. Live mode then fails gracefully —setMode("live")catches the error, raises anmc-banner-errbanner + error toast, and falls back to snapshot.
Run, test, build
pnpm install
# refresh the bundled snapshot from demo.flow-master.ai
pnpm fetch:scenarios
# dev (with backend proxy for live mode)
pnpm dev # → http://127.0.0.1:5173
# tests
pnpm test # vitest, 22 tests across api, live,
# synthetic, layout, store
# build
pnpm build # tsc + vite, single chunk ~225 KB gz
# end-to-end smoke + screenshots
pnpm qa:smoke # playwright headless, 28 assertions
# DOM layout audit
pnpm qa:layout
File map
src/
├── data/ # ProcessScenario domain + snapshot + blueprint catalog
├── lib/ # API client + live-scenario builder (+ tests)
├── state/store.ts # zustand: scene, mode, scenario, tour, recents, toasts
├── graph/layout.ts # dagre LR auto-layout (+ tests)
├── components/ # ProcessGraph, Inspector, LeftRail, CommandBar, Tour,
│ # Telemetry, Toaster, icons
├── scenes/ # Landing, MissionControl, RunHistory
├── App.tsx # shell: topbar (mode pill / refresh / tour / ⌘K)
├── index.css # design system (~700 lines)
├── main.tsx
└── scenarios.json # cached snapshot of demo.flow-master.ai
qa/
├── smoke.mjs # Playwright e2e + 9 screenshots
└── layout_audit.mjs # programmatic clipping/overlap check
vite.config.ts # /api → demo.flow-master.ai proxy for dev
fetch_scenarios.mjs # Node script to refresh src/scenarios.json
Deploy
Production deployment is tracked in
FM06/flowmaster-ops PR #1164
(merged), which added three resources to manifests/overlays/demo/:
mc-deployment.yaml— 2-replica nginx Deployment in thedemonamespacemc-service.yaml— ClusterIP service on port 80mc-ingress.yaml— Traefik ingress atcanvas.flow-master.aiwith cert-manager DNS-01 cert
Cloudflare A record canvas.flow-master.ai → 65.21.71.186 was added at the
same time. The cluster certifies with cert-manager (Let's Encrypt DNS-01)
and Traefik fronts the nginx pods. nginx reverse-proxies /api/* to
https://demo.flow-master.ai so the SPA is same-origin (no CORS).
pnpm build # builds dist/
docker build -t gitea.flow-master.ai/shad/mission-control-demo:sha-<git> .
docker push gitea.flow-master.ai/shad/mission-control-demo:sha-<git>
# Then bump the image pin in manifests/overlays/demo/mc-deployment.yaml.
What's intentionally not here
- Authentication UI (dev-login is used because this is a demo lane).
- Mutation endpoints (every action is preview-only and the toast names the endpoint).
- Mobile layout (1440px-and-up demo surface).
- Route persistence / deep linking (scene + scenario live in memory).
- i18n (English only).
Small, well-scoped follow-ups — not architectural changes.