Pulls the actual FM06/flow-master-design-philosophy doctrine
(DESIGN_PHILOSOPHY.md + SYNTHESIS.md + IMPLEMENTATION_STANDARD.md
+ ADR 0001/0002) and rebuilds the canvas to match: 'operations
cockpit', 'industrial, instrumented, accountable'. Light paper
canvas + navy frame + amber accent + 1px rules + square edges +
monospace operational labels.
WHAT CHANGED
- ProcessGraph.tsx rewritten: square 1px navy nodes, mono uppercase
labels, orthogonal step edges (ReactFlow type:'step' → MLHV only),
two-layer Background (8px minor + 64px major navy hairline grid),
doctrinal palette tokens via var(--bp-*).
- BlueprintFrame.tsx (new): top instrument readout strip
(DEF / VERSION / HUB / NODES / EDGES / SRC / MODE / TX), top + left
rulers with 8/64px ticks, navy corner glyph at origin, bottom
status legend.
- index.css: scoped [data-canvas="blueprint"] block (~280 lines)
declaring 11 doctrinal hex tokens once; opacity derivatives go
through CSS color-mix(in srgb, var(--bp-navy) 13%, transparent)
not raw rgba(). No box-shadow on selected node (outline instead).
- LeftRail.tsx: gate toast now names the real endpoint
(POST /api/runtime/transactions/{id}/actions/{submit,save_draft})
on the unsigned-in path too, restoring the convention from the
prior real-mutations pass.
- qa/smoke.mjs updated for the new selectors (.bp-node,
.bp-readout-blueprint). Old guided-tour assertions replaced with
Studio scene assertions. 27/27 PASS.
- qa/smoke_blueprint.mjs (new): 15 assertions covering S1–S3 + S5.
- qa/palette_audit.mjs (new): three checks — doctrinal CSS hex,
no raw rgba() in blueprint scope, no hardcoded color literals in
blueprint TSX. All pass.
ORACLE-REVIEWED
Round 1 FAIL: hardcoded TSX color literals + box-shadow + rgba +
narrow palette audit. Round 2 PASS after fixing all four.
CONTRACT EVIDENCE
- vitest: 5 files, 24 tests, all green
- main smoke: 27/27, 0 console errors
- blueprint smoke: 15/15, 0 console errors
- palette audit: 11 CSS doctrinal hex tokens, 0 raw rgba, 2 TSX files clean
- vite build: green
Confidence: high
Scope-risk: narrow (scoped [data-canvas="blueprint"])
Not-tested: pixel-level visual diff (Oracle could not inspect images
this round; relied on programmatic DOM + path-command assertions)
FlowMaster — Mission Control demo
A polished, presenter-ready command-center for FlowMaster. Lives at
shad/flowmaster-mission-control-demo
on Gitea.
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 themc.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,
which adds 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 atmc.flow-master.aiwith cert-manager DNS-01 cert
Status: the PR is open and mergeable. The live URL https://mc.flow-master.ai is not yet serving — two pre-merge action items remain for the trusted updater:
- Cloudflare A record
mc.flow-master.ai → 65.21.71.186, 91.98.159.56(same ashakeem.flow-master.ai). - Gitea Actions must have published the first image tag (
gitea.flow-master.ai/shad/mission-control-demo:sha-…) —mc-deployment.yamlpins that explicit SHA, not:latest.
Until the PR merges + DNS is added, the artifact is this repo + the open PR.
pnpm build
# dist/ is the static site. The Dockerfile bakes it into a tiny nginx
# image. CI in .gitea/workflows/build.yml builds + publishes on push to main.
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.