All notable changes to this project are documented here.
The format is based on Keep a Changelog, and this project adheres to Semantic Versioning.
0.8.0 - 2026-05-20
The rebuilt-core release: everything is a plugin, headless by default, plugins create plugins. Open Design's research-preview architecture has been replaced with a small, boring engine plus a plugin surface — design systems, slices, prototypes, exports, and Figma itself all live in plugins now. The desktop app is a thin wrapper around the OD CLI, so the same engine runs in Claude Code, OpenClaw, Hermes Agent, and chat bots in Lark / Discord / Slack. Critique Theater matures through Phase 16 (rollout ratchet, conformance API, 9 Prometheus metrics, Grafana dashboard, M0 dark-launch by default). 149 design systems now ship with structured tokens.css + components manifests across 60+ new brand fixtures. Italian (it) locale + CJK font fallback. New media providers: Leonardo.ai, ElevenLabs, SenseAudio. Packaged auto-update lands on both macOS and Windows, battle-hardened through the preview cycle. Plus a top-to-bottom visual refresh, Quick-brief discovery overhaul, PostHog v2 analytics schema, manual edit UX overhaul (focus mode, uploads, remove-element patch), custom CLI agent profiles, and HTML Anything landing page. 305 merged PRs by 75 contributors since 0.7.0.
- Plugin engine rebuild with
packages/plugin-runtime,packages/registry-protocol, andpackages/host— the engine surfaces the plugin lifecycle through a small, neutral API so design systems, slices, prototypes, exports, and even Figma itself can live as plugins. - Plugin registry detail drawer with trust badges and marketplace metadata. (#2087)
- GitHub rate-limit fallback for marketplace plugins keeps install / refresh flows reliable when GitHub API is throttled. (#2064)
- Plugin Publish-repo flow creates the author's repo correctly. (#2332, #2363)
- CLI plugin publish reads manifest version when the stored row is the
0.0.0sentinel. (#1903) - Block raw publish CLIs from the authoring summary — keep agents on the OD publish path. (#2380)
- Demote Plugins + Integrations to the nav rail footer so primary surface stays focused. (#1806, #2360, #2397)
- Phase 9 — drop-in mount wrapper, native i18n for
de/ja/ko/zh-TW. (#1315) - Phase 10 — daemon adapter conformance lab + degraded registry. (#1316)
- Phase 11 — Playwright stage suite (happy path, interrupt, 3 viewports, a11y). (#1317, #1483)
- Phase 12 — 9 Prometheus metrics + 6 log events + OTel span + Grafana dashboard. (#1485)
- Phase 13 — reducer p99 benchmark + surface coverage walker. (#1318)
- Phase 15 — rollout resolver + Settings toggle hook. (#1320)
- Phase 16 — M-phase rollout ratchet +
/api/critique/conformance. (#1499) - Wireup with M0 dark-launch by default. (#1338)
- Settings toggle with dedicated section + i18n keys across 6 locales. (#1484)
- Token channel default-on (PR-D) so the new fixture pipeline is the default surface. (#1544)
- Structured
tokens.cssfor 60+ new brands across AI, devtool, SaaS, fintech, docs, consumer, hardware, cultural categories (Apple, Stripe, Airbnb, Vercel, Notion, Linear, GitHub, Figma, Slack, Discord, OpenAI, Shopify, Spotify, Uber, Cursor, and many more). (#1652, #1794, #1841, #2023, #2028, #2029, #2033) - Token fixture catalog — 20 brand + 20 product + remaining style fixtures, component-fixture coverage report. (#2037, #2040, #2043, #2049)
- Component manifests — extract + consume manifests for design systems. (#2051)
- Import design-system projects via the discovery flow.
- Perplexity design system. (#1747)
- Local custom CLI agent profiles for arbitrary CLI agents. (#378)
- Leonardo.ai image provider. (#1123)
- ElevenLabs audio support. (#1384)
- SenseAudio TTS provider + BYOK chat with image / video generation tools. (#1633, #2065)
- User-configurable model alias for the media dispatcher. (#1277)
- Cursor Agent live model id parsing + auth diagnostics. (#1538, #2228)
- Manual edit UX overhaul — focus mode, inline uploads, remove-element patch. (#1516)
- Manual edit inspector. (#1448)
- Tweaks toolbar bound to the artifact panel (toggle visibility from the panel chrome).
- Custom select primitive for cleaner dropdowns.
- Collapsible comment side panel.
- Export as image in the share menu.
- Render GFM tables in markdown artifacts and chat.
- Surface saved Project instructions for review and retrieval.
- Copy-to-clipboard for user messages.
- Filter-by-kind dropdown on the design-files viewer.
- Quick-brief: collapse freeform clarification into a single form. (#2226)
- Plugin inputs as authoritative Quick-brief answers. (#2243)
- Stabilize discovery brand answers in prompts. (#1861)
- Daemon surfaces discovery form answers to agents. (#2071)
- Packaged auto-update for both macOS and Windows. (#2362, #2270, #2403)
- Updater hardening through the preview cycle — release validation, deferred installer on Windows, applied-state clearing, download / install handoff hardening, smoke-recovery. (#2565, #2575, #2592, #2595, #2677, #2687, #2700)
- Desktop updater UI flow — new in-app updater popup.
- Packaged update apply observations captured for telemetry / debugging. (#2429)
- Nightly + preview package identity so beta installs don't collide with stable. (#2437)
- macOS Dock icon stays put when desktop-pet window opens. (#2413)
- Refresh Open Design app visuals — new app icons, logo, brand glyphs. (#2436)
- Linux packaged client parity smoke coverage.
- Ensure node binary dir is on PATH for agent sub-processes on Windows. (#1989)
- Italian (it) locale — full UI translation, brings supported languages to 19. (#1323)
- CJK font fallback for Chinese / Japanese / Korean. (#2227)
- Refresh + polish French UI locale.
- Translate template platform selection + Companion surfaces to Chinese. (#1491)
- Localize accent controls in settings, comment-panel strings (#1390, #1392), and skill validation messages.
- PostHog v2 event schema. (#2285)
- Unify
page_name+ onboarding / design-system page_views. (#2390) - Upgrade
posthog-node4 → 5 in the daemon. (#2309) - One-click log export from Settings → About.
- HTML Anything page + responsive landing header. (#2452)
- Rebuild
/templatescatalog fromdesign-templates. (#2369) - Refresh templates + add tutorials channel on the landing site. (#2409)
- Blog routes on the landing site.
- Search Console reporting workflows + GSC report opportunities. (#2388)
- WeRead year-in-review HyperFrames template.
- Critique Theater dark-launched at M0 by default, gated through the new rollout ratchet so phases can be promoted independently.
- Plugin trust badges unified across registry surfaces.
- Plugins and Integrations moved to the nav rail footer (#1806, #2360, #2397) — keep primary surface focused.
- Block pitch-deck placeholder publishes and unbreak framework decks.
- Rename FileViewer "Share" button to "Export".
- Confirm before deleting a saved template in New Project.
- Restore consistent app header layout on the entry view. (#1519)
- Refine preview and project dropdown controls. (#1514)
- Pin chat during content growth.
- Auto-scroll feedback form.
- Routines history rows deep-link to their specific conversation. (Fixes #1505)
- Hide resolved comments from preview overlays.
- Keep filter pill hover labels readable.
- Improve replace-modal button hover contrast.
- Freeze completed run durations across conversations.
- Align Home prompt overlay with textarea so caret lands on click.
- Restore release-light background. (#1540)
- Allow downloads from preview iframes; fall back to srcDoc when HTML preview needs sandbox shim.
- Coalesce chokidar rewrite bursts before refreshing files.
- Reveal memory editor after edit click; distinguish expanded memory preview action.
- Auto-annotate imported HTML elements for Tweaks selection. (#892)
- Stable shared frame screen paths from referrer.
- Restore custom dropdown chevron for timezone selector in dark mode.
- Daemon run recovery across reloads. (#2374)
- macOS Dock icon stays put when desktop-pet window opens. (#2413)
- Align Windows smoke update root with portable installs. (#2376)
- Nightly release smoke identity. (#2446)
- Improve desktop updater ready UI. (#2403)
- Forward proxy env vars to packaged sidecars.
- Detect mise-installed npm package bins.
- Launch Windows updater fixture via Node. (#2364)
- Desktop "Export PDF" opens a direct "Save as PDF" file dialog and writes the PDF to disk, instead of opening the macOS system print dialog. (Fixes #1774)
- macOS close exits fullscreen before hiding.
- Daemon's external-browser opener fixed on Windows.
- Surface discovery form answers to agents. (#2071)
- Stabilize discovery brand answers in prompts. (#1861)
- ACP model detection timeout is configurable.
- Wrap Claude smoke test stdin as stream-json.
- Preserve Claude tool inputs. (#1476)
- Codex CLI path fallback UX. (#1205)
- Treat Codex reconnect events as warnings, not fatal errors. (#1482)
- ACP config options used for model selection. (#1208)
- Remove OpenCode stdin dash sentinel; soft empty API response handling.
- Forward external MCP servers to OpenCode.
- Critique Theater Phase 14 user guide + 2 AGENTS module maps. (#1319)
- Windows native setup notes in
AGENTS.md. - Comprehensive contributor guide in
TRANSLATIONS.md. - RTL_LOCALES UI guidance +
es-ESalignment. - Sync
zh-TWREADME with the English version. - Sync Windows troubleshooting link across locale READMEs.
- Refresh contributors wall + GitHub metrics SVG.
- Clarify Intel Mac ZIP packaging support (includes the Monterey verified path and the Finder
PATHcaveat for packaged CLI detection). (Fixes #327) - README inventory badges sync — skills 31 → 131, design-systems 72 → 149. (#1899)
- 0.8.0-preview banner + Discussion #1727 pointer. (#1781)
- Active 0.8.0 contributors point at
main. (#1846)
- Critique Theater Playwright stage suite (happy, interrupt, 3 viewports, a11y). (#1317, #1483)
- Reducer p99 bench + surface coverage walker. (#1318)
- Harden e2e extended coverage state assertions. (#2245)
- Visual regression PR workflow (CI).
- Component manifest extraction + daemon consume path. (#2051)
- OD CLI wraps GitHub CLI (so plugins create plugins).
pnpm i18n:coverageinformational report.- Issue templates: bug, feature, preview/v0.8.0 + chooser config. (#1708)
0.7.0 - 2026-05-12
A memory-plus-UI release: auto-memory store carries agent context across runs and projects, Critique Theater advances to Phase 7 (state machine + replay) with daemon-side Phase 6.2 artifact extraction, HyperFrames lands HTML-in-Canvas end-to-end, and the web UI gets a top-to-bottom Designs tab redesign, in-context preview comments, a unified Media tab, and a tweaks palette with HSL hue-shift recoloring. Plus responsive design handoff outputs, install/uninstall skills & design systems in-app, HTTP 206 range requests for video/audio, scheduled routines for unattended agent runs, macOS Intel (x64) builds, an official Nix flake, four new design systems (hud, loom, trading-terminal, WeChat), and an agent-browser skill. 107 merged PRs since 0.6.0.
- Phase 7 — web client state machine. Reducer plus
useCritiqueStream/useCritiqueReplayhooks so the critique UI replays cleanly and reconnects mid-run. (#1307) - Phase 6.2 — daemon artifact extraction + endpoint. Critique-tagged artifacts are extracted server-side and exposed via a dedicated endpoint. (#1085)
- Designs tab redesign — cards with covers, tags, overflow menu, and multi-select. (#1161)
- In-context comment thread for the artifact preview — comment directly on a preview without leaving the workspace. (#1276)
- Unified Media tab — Image / Video / Audio entries consolidated into a single tab. (#1167)
- Tweaks palette popover with HSL hue-shift recoloring — adjust hue/saturation/lightness inline. (#1292)
- Responsive preview + design handoff outputs — tablet/mobile preview auto-fit, 2025–2026 responsive viewport matrix, screen-file-first export policy, DESIGN-HANDOFF / DESIGN-MANIFEST exports for coding-tool implementation. (#1224)
- Thumbs-up / thumbs-down feedback widget under completed assistant turns. (#1308)
Cmd/Ctrl+,opens Settings with a platform shortcut badge in the menu. (#1173)- Finalize design package + Continue in CLI buttons. (#974)
- Fetch models button for BYOK providers so newly added models discover themselves on demand. (#1034)
- Provider-model fetch results sorted alphabetically. (#1097)
- Collapsible MCP JSON field-mapping helper. (#1136)
- Question forms scroll to top of viewport instead of pinning to the bottom. (#1044)
- Sidebar tabs flush to the workspace edge with internal scrolling. (#1038)
- Design file rename support. (#894)
- Auto-memory store with chat-protocol-aware extraction. Agents accumulate durable context across runs and projects. (#999)
- Install / uninstall for skills & design systems directly from the desktop app. (#1003)
- HTTP 206 range request support for video / audio files (closes #784). (#1105)
- Scheduled routines for unattended agent runs. (#1033)
- Guard against agent-emitted stub artifact regressions. (#1171)
- Reject filesystem root folder imports. (#1266)
- Split agent runtime definitions into per-runtime modules. (#1063)
- Split route registration into focused modules. (#1043)
- HTML-in-Canvas lands across web + skills. Embedded HTML renders inside canvas-mode templates end-to-end. (#866)
- Generic skills + skills/design-templates split + finalize-design API. Skill model refactor that separates code-driven skills from design-template assets. (#955)
- Reliable
agent-browserskill. (#1284) - WeChat design system +
login-flowskill (also fixes API-modetool_callsbug). (#1083) - Three new design systems:
hud,loom,trading-terminalwith locale coverage. (#1069) release-notes-one-pagerskill with supporting docs. (#873)- Improve design files grouping. (#1082)
- Replace time-specific Orbit greetings with neutral defaults. (#1291)
tokens.cssschema for design systems (default+kami). (#1231)
- Install onboarding links for unavailable local CLIs. (#985)
- macOS Intel (x64) build support in release workflows. (#759)
- Official Nix flake with home-manager and NixOS support. (#402)
- Beta release packaging cache optimizations. (#1095)
- Traditional Chinese QUICKSTART + Chinese doc links fixed. (#753)
- Conversation run isolation enforced — concurrent runs no longer cross-contaminate state. (#1271)
- Default English resource i18n fallback so missing translations don't break agent prompts. (#1270)
[codex]Claude Code exit diagnostics improved. (#1267)[codex]empty API responses handled as no output (not as errors). (#1244)- Codex CLI path fallback UX improvements. (#1205)
- Persist Appearance accent color selection so swatch picks survive Settings close. (#1439)
- Load Orbit template choices from
design-templatesinstead ofskills(aligns with the skill model refactor). (#1442) - Restore custom dropdown chevron for the timezone selector in dark mode. (#1368)
- Polish EntryView UI — sidebar layout, folder tabs, slim form, blue selected token. (#1360)
- Translate Design Files refresh strings instead of hardcoding English. (#1300)
- Pretty-print JSON file previews. (#1206)
- Keep Tweaks selection usable without annotations. (#1268)
- Render static previews for sketch JSON files. (#1060)
- Ignore
<artifact>tags inside markdown code spans and fences. (#1132) - Refresh home projects after deleting a conversation. (#1219)
- Complete finished tool calls missing results. (#1240)
- Intercept prose-as-HTML artifacts before they hit disk. (#1144)
- Truncate entry footer pet label. (#1150)
- Surface connector auth errors and stop silent popup close. (#1128)
- Center close button in MCP picker dialog. (#1137)
- Prevent chat messages overflowing into the workspace area. (#1104)
- Prevent creating a new conversation when the current one is empty. (#1086)
- Dispatch Examples preview on
od.preview.type. (#1001) - Scroll Settings content back to top on section change. (#997)
- Allow pod-to-chat comment text to wrap instead of truncating. (#1156)
- Localize MCP server settings panel for Chinese UI. (#760)
- Restore media config from daemon on startup. (#687)
- Suppress autosave indicator for draft-only Connector key edits. (#1232)
- Fix link handling in example preview iframe sandbox. (#701)
- Handle popup-blocked PDF export with native Electron print dialog. (#973)
- Translate comments panel UI to Chinese. (#1139)
- Hide surface filter tabs with zero count in Design Systems view. (#965)
- Prevent design system filter popover from shifting position on reopen. (#960)
- Exit fullscreen before hiding window on macOS close. (#1249)
- Enforce minimum window size on main client. (#1203)
- Allow
about:blankpopup for PDF export fallback. (#1081) - Fix daemon browser opener on Windows. (#953)
- Wire finalized assistant message writes to the Langfuse report bridge so reports settle reliably. (#1402)
- Remove OpenCode stdin dash sentinel. (#1365)
- Use ACP config options for model selection. (#1208)
- Persist
runStatus/endedAton chat run termination. (#1230) - Prefer
opencode-clioveropencodebinary so OpenCode Desktop installs resolve to the CLI. (#818) - Support OpenCode Write tool display as card. (#1126)
- Pin API-mode override above discovery layer. (#1207)
- Close running app before silent reinstall on Windows. (#1238)
- Build desktop before packaged typecheck. (#1093)
- Prevent project type selector arrows from overlapping tabs. (#1091)
- Increase top padding in modal footer for better visual balance. (#957)
- Clear stale upload failure banner when previewing files. (#797)
- Remove Trump pet from bundled community pets. (#1103)
- Add "When NOT to emit" guardrail to artifact handoff prompt. (#1145)
- Pending prompt clearing for templates. (#1148)
- Set writable
OD_DATA_DIRdefault fornix run. (#1159) - Landing-page SEO: correct canonical, add
robots.txt+ favicons. (#1061)
MAINTAINERS.md+ CONTRIBUTING entry-point for maintainer rules. (#1290)- Traditional Chinese QUICKSTART + Chinese doc link fixes. (#753)
-
Stabilize extended Playwright coverage. (#1341)
-
Expand nightly UI and desktop regression coverage. (#1256)
-
Harden e2e smoke and release reports. (#1140)
-
Expand entry and settings automation coverage. (#954)
-
Refreshed generated GitHub metrics SVG and contributors wall. (#1115, #1117, #1183, #1188, #1328, #1330)
-
Plugin & marketplace system — Phase 2A + 1 + 1.5 + 2B + 2C entry slice + 3 (full) + 4 (full incl. OD_BUNDLED_ATOM_PROMPTS default ON) + 5 (full incl. live S3 impl; postgres adapter still stubbed) + 6 (full incl. asset rasterisation) + 7 (all six code-migration atom impls landed; full pipeline e2e covered by smoke test) + 8 (full incl. GenUI \u2192 decision bridge + handoff promotion ladder bridge) + bundled scenarios + bundled-scenario fallback resolver. Spec:
docs/plugins-spec.md. Living plan:docs/plans/plugins-implementation.md.od plugin events purgeadmin escape hatch (Phase 4). NewpurgePluginEventBuffer()returns pre-purge stats{ purged, firstId, lastId, preNextId }so an operator can audit what was discarded. Loopback-onlyPOST /api/plugins/events/purgeroute +od plugin events purge --confirmCLI subcommand (refuses to run without--confirmso a stray invocation never drops audit data accidentally).od plugin manifest <id>+od plugin sources(Phase 4). New CLI subcommands for plugin authors and ops:od plugin manifest <id>prints just the parsed manifest JSON (no wrapper, no fsPath / installedAt noise) so authors can diff against their on-diskopen-design.json.od plugin sourceslists every distinct(sourceKind, source)tuple + plugin count, sorted by descending count then alphabetically, with the per-bucket plugin list sorted by id. NewpluginSourceBuckets()pure helper backs the CLI for byte-deterministic output.od plugin events snapshot/stats+ tail filters (Phase 4). Extends §3.II1 with:GET /api/plugins/events/snapshotfor non-SSE one-shot reads (dashboards that don't want a live connection);GET /api/plugins/events/statsreturns asummarisePluginEvents()rollup (counts byKind, byPluginId — skipping empty ids — plus oldest/newest at + id range);--kind <k>and--plugin-id <id>filter flags work on bothod plugin events tail(client-side post-render) and the newod plugin events snapshotsubcommand. CLI pretty-prints the stats rollup with sorted-key counts for byte-determinism.- More plugin event producer hooks (Phase 4). Extends §3.II1 with:
installPluginacceptseventKind: 'installed' | 'upgraded'so the upgrade route distinguishes the operation in the live tail;POST /api/plugins/:id/trustemitsplugin.trust-changed;POST /api/applied-plugins/pruneemitsplugin.snapshot-prunedwhen anything was actually removed;POST /api/marketplaces/:id/refreshemitsplugin.marketplace-refreshed. Each hook is best-effort and never blocks the underlying mutation if the ring buffer throws. - Plugin event ring buffer + SSE tail (Phase 4). New
apps/daemon/src/plugins/events.tsships an in-memory FIFO ring buffer (capped at 1000 entries, monotonic ids, fan-out subscribers) for plugin lifecycle events:plugin.installed/.upgraded/.uninstalled/.trust-changed/.applied/.snapshot-pruned/.marketplace-refreshed. Producer hooks landed on the installer (install + uninstall). NewGET /api/plugins/eventsSSE route emits the backlog on connect (with optional?since=<id>trim) then forwards live events. CLI:od plugin events tail [-f] [--since <id>] [--json]— non-follow mode drains backlog + exits;-fkeeps the stream open for ops dashboards. od plugin doctor --strict+ verify strict propagation (Phase 4). New--strictflag onod plugin doctorpromotes warnings to failures (exit 4 distinguishes 'strict failed' from doctor errors at exit 1). TheverifyPlugin()orchestrator gains a matchingstrict: trueconfig knob that flows through.od-verify.jsonso plugins can lock 'no warnings allowed' as a one-line CI policy.od daemon db verifySQLite integrity check (Phase 5). NewverifySqliteIntegrity()pure helper wraps PRAGMAintegrity_check(orquick_checkwith--quick) + PRAGMAforeign_key_check. Returns a structured{ ok, mode, issues[], elapsedMs, generatedAt }report with issues taggedkind='integrity' | 'foreign_key'. Loopback-onlyPOST /api/daemon/db/verifyroute +od daemon db verify [--quick]CLI subcommand — exit 0 on ok=true, 4 on any issue, so CI can wire it into a pre-deploy check.od daemon db vacuum(Phase 5). New loopback-onlyPOST /api/daemon/db/vacuumruns SQLite VACUUM and reports before/after sizes + reclaimed bytes + elapsed ms. Useful after large delete batches (snapshot prune, plugin uninstall) shrink rows but leave space allocated to the file. CLI:od daemon db vacuum [--json].od daemon db statusSQLite inventory (Phase 5). NewinspectSqliteDatabase()pure helper +GET /api/daemon/dbroute returns a structured report:kind('sqlite'), file location, size on disk (primary + WAL + SHM), schema version (user_versionPRAGMA), and per-table row counts (system tables excluded, lexicographic order). CLI:od daemon db status [--json]lets ops sanity-check deployments at a glance + compare expected-vs-actual table rosters across daemon upgrades.od plugin verify <id>CI meta-command (Phase 4). NewverifyPlugin()pure orchestrator aggregatesdoctor+simulate+canon --checkinto one pass/fail report. Reads<plugin-folder>/.od-verify.json(or--config <path>) so plugin authors commit their CI checks into their repo. Each check resolves topassed | failed | skipped | unsupported; aggregate passes iff every enabled check is passed or skipped (unsupportedbubbles up as a fail to keep CI honest). One-liner CI workflow:od plugin verify my-plugin— exit 0 on pass, 4 on fail, 2 on CLI/config error.od plugin simulate <id>pipeline dry-run (Phase 4). NewsimulatePipeline({ pipeline, signals, iterationCap? })pure helper walks every stage in a plugin's pipeline against caller-supplied signals (constant snapshot OR per-iteration generator function) and reportsoutcome ∈ { single | converged | cap | unparsable }per stage plus aggregateoutcome ∈ { all-converged | all-single | mixed | cap-hit | unparsable }. CompanionparseSignalKv()parses repeatable-s key=valueCLI flags into the closedUntilSignalsvocabulary with typo guards. CLI:od plugin simulate <pluginId> [-s key=value ...] [--cap <n>] [--json]— exit 4 on cap-hit/unparsable so CI can hook this into a pipeline check.od plugin statsinventory health report (Phase 4). NewpluginInventoryStats()+snapshotInventoryStats()pure helpers aggregate installed-plugin counts (bysourceKind/trust/taskKind, bundled vs. third-party split, plugins with elevated capabilities —fs:write/subprocess/bash/network/connector:*) and snapshot health (status breakdown, project / run linkage, oldest / newest applied timestamps). NewGET /api/plugins/statsroute +od plugin stats [--json]CLI subcommand for at-a-glance fleet audit.od plugin canon --check <expected-file>byte-equality fixtures (Phase 4). New--checkmode onod plugin canoncompares the canon output against an on-disk fixture and exits 4 on mismatch with a per-line diff preview. Lets plugin authors commitrenderPluginBlock()regression fixtures into their owntests/without writing a fresh test harness.od plugin canon <snapshotId>show prompt block (Phase 4). NewGET /api/applied-plugins/:snapshotId/canonroute renders the canonical## Active plugin/## Plugin inputsblock this snapshot splices into the system prompt, byte-equal to what the agent reads viacomposeSystemPrompt. Plain-text response when the request sendsAccept: text/plainso shell pipes work cleanly; JSON wrapper otherwise. CLI:od plugin canon <snapshotId> [--json].od plugin snapshots show / diffdebugging (Phase 4). NewGET /api/applied-plugins/<id>-backedod plugin snapshots show <snapshotId>dumps a fullAppliedPluginSnapshotfor inspection. NewdiffSnapshots()pure helper +od plugin snapshots diff <a> <b>compares two snapshots field-by-field with adigestEqualflag that surfaces the e2e-2 invariance check at-a-glance. Compares identity, inputs map, capabilities, resolvedContext, connectors, mcpServers, genuiSurfaces, pipeline (stages + per-stage atoms / until), assets. Entries sort lexicographically so output is byte-deterministic.od plugin diff <a> <b>author tooling (Phase 4). NewdiffPlugins({ a, b })pure-helper compares twoInstalledPluginRecordvalues and returns a structured{ pluginId?, entries[], added, removed, changed }report. Compares top-level fields (id, version, sourceKind, source, trust, capabilitiesGranted) plus manifest body (title / version / description / license / tags / kind / taskKind / mode / capabilities / inputs / context.skills / context.craft / context.assets / pipeline.stages + per-stage atoms / connectors.required / genui.surfaces). Collection diffs collapse to<n> added, <m> removedsummaries; entries sort lexicographically by field path so the report is byte-deterministic across re-runs. CLI:od plugin diff <id-a> <id-b> [--json]renders +/-/~ glyphs.od atoms info <id>+ atom catalog drift fix (Phase 4). NewGET /api/atoms/:idreturns catalog metadata + the bundled SKILL.md body. Catalog promoted nine atoms (code-import,design-extract,figma-extract,token-map,rewrite-plan,patch-edit,diff-review,handoff) fromstatus='planned'to'implemented'to reflect the daemon impls landed across §3.N–§3.S;build-testadded as a missing catalog row. Net: zeroplannedatoms remaining.od plugin upgrade <id>re-install from recorded source (Phase 4). NewPOST /api/plugins/:id/upgraderoute streams the same SSE shape as/install, internally callinginstallPlugin()against the plugin's recordedsourcestring (so the github / https / local-folder byte path replays end-to-end). 409s withcode='bundled-plugin'when the plugin shipped bundled with the daemon (those upgrade with the daemon image), andcode='missing-source'when the recorded source is empty. CLI:od plugin upgrade <id>— exit 0 on success, 1 on installer error.- patch-edit atomic file writes (Phase 7 safety patch). Every
fs.writeFileinsidepatch-edit(file content +plan/steps.json+plan/receipts/) now routes through anatomicWriteFile()helper that writes to a sibling tmp file and renames into place via POSIXrename(2). A daemon crash mid-write can no longer leave the source file truncated; rejection-path failures (context mismatch, etc.) leave the original byte-equal. Cleanup path unlinks the orphan tmp on failure so disk noise stays bounded. od plugin search <query>+ filters onod plugin list(Phase 4). NewsearchInstalledPlugins({ plugins, query?, taskKind?, mode?, tag?, trust?, bundled? })pure-helper ranks free-text matches across id / title / description / tags (exact wins over substring; tag-exact ranks ahead of title-substring) and AND-combines with structural filters. CLI gainsod plugin search <query>and adds--task-kind / --mode / --tag / --trust / --bundled / --no-bundledknobs tood plugin list. Search results showmatched=[…]per row so the user can trace which field caught the hit.od plugin pack <folder>distribution archive (Phase 4). NewpackPlugin({ folder, out? })helper builds a gzip-compressed tar archive of an author's plugin folder, ready to install via the installer's HTTPS-tarball path. Default output path is<basename>-<manifest.version>.tgzbeside the folder. Skiplist matches the installer's extract-time exclusions (node_modules,.git,dist,build,out,coverage,.turbo,.cache,.pnpm-store,.DS_Store, etc). Symlinks rejected at both walk and tar-filter passes so what authors pack matches what the installer accepts. The output archive itself is always excluded from its own contents (no spiral when--outlands inside the folder).od plugin validate <folder>author-side lint (Phase 4). Pre-install lint pass against an unfinished plugin folder. NewvalidatePluginFolder({ folder, registry? })helper reads the folder via the sameresolvePluginFolder()the installer uses (manifest parsing is byte-equal to install time) and runsdoctorPlugin()against the supplied registry view. Surfaces atom-id mismatches, manifest parse errors, unparseableuntilexpressions, and unresolved skill / DS / craft refs without dirtying the registry table. CLI takes--no-daemonfor fully-offline runs and--jsonfor machine-readable output. Exit code4whendoctor.ok=false,2on CLI/folder errors,0on clean.OD_BUNDLED_ATOM_PROMPTSdefault flipped to ON (Phase 4). With every Phase 6/7/8 atom impl shipping its own SKILL.md fragment (build-test, code-import, design-extract, figma-extract, token-map, rewrite-plan, patch-edit, diff-review, handoff), the audit-block on flipping the default is lifted. Runs with a plugin snapshot now ALWAYS get the matching## Active stageblocks spliced into the system prompt without operators having to set the env var. SetOD_BUNDLED_ATOM_PROMPTS=0to opt out (snapshot replay against pre-flip daemons, regression bisects that need byte-equal prompts). Runs without a snapshot stay byte-equal so the change is invisible outside the plugin path.S3ProjectStoragelive implementation via AWS SigV4 (Phase 5). Newapps/daemon/src/storage/aws-sigv4.tsships a minimal AWS Signature V4 signer using onlynode:crypto(no@aws-sdk/*dependency \u2014 keeps the daemon shippable as a small binary). The signer is verifiably correct against the AWS-published reference vector (GetObject example).S3ProjectStoragenow implements all fiveProjectStorageops (read / write / list / delete / stat) viaglobalThis.fetch, with virtual-host-style URLs by default and path-style forOD_S3_ENDPOINToverrides (Aliyun OSS, Tencent COS, Huawei OBS, MinIO). LIST walksNextContinuationTokenfor paginated buckets. Credentials read fromOD_S3_ACCESS_KEY_ID/SECRET/SESSION_TOKENfirst, falling back toAWS_*env vars so existing IAM-role /aws configuresetups drop in unchanged.runHandoffAtom()pipeline-driven bridge (Phase 8 entry slice). New helper reads the canonical state previous atoms wrote (<cwd>/review/decision.jsonfrom diff-review,<cwd>/critique/build-test.jsonfrom build-test) and returns the updatedArtifactManifestwith the righthandoffKind/exportTargets[]attached. Promotion ladder (spec §11.5.1):reject\u2192design-only;accept/partialno-build-test \u2192implementation-plan; +build.passing\u2228tests.passing\u2192patch; + both signals + docker/cli export \u2192deployable-app. Monotonicity enforced viarecordHandoff(); a manifest carrying'patch'won't demote to'design-only'even when a follow-up reject arrives.runAndPersistHandoff()+ auto-handoff from diff-review GenUI bridge (Phase 8 entry slice). New on-disk shell round-trips<cwd>/handoff/manifest.json: reads existing manifest (or falls back tomanifestSeed, then to a minimal default), callsrunHandoffAtom(), and writes back only when something changed (persistMode: 'created' | 'updated' | 'skipped'). The diff-review GenUI bridge now auto-invokes this after recording the user's decision, so the Phase 8 promotion ladder closes end-to-end without an agent turn: user clicks Accept all in the web composer → daemon writesreview/decision.jsonANDhandoff/manifest.jsonwith the righthandoffKindset; failure surfaces on the bridge result ashandoffErrorwithout regressing the diff-review write.- figma-migration pipeline e2e smoke (Phase 6). New
plugins-figma-migration-e2e.test.tswalks figma-extract (stubbed REST) → token-map → diff-review → handoff end-to-end. Locks the scenario folder roster (plugins/_official/scenarios/od-figma-migration/open-design.jsonstages: extract → tokens → generate → critique). - Full code-migration pipeline e2e smoke test (Phase 7-8). First integration test that walks every code-migration atom (code-import \u2192 design-extract \u2192 token-map \u2192 rewrite-plan \u2192 patch-edit \u2192 build-test \u2192 diff-review \u2192 handoff) on a Next.js fixture repo without any agent in the loop. Locks the inter-atom file contract so a future PR can't break the chain by silently renaming
code/tokens.jsonor adding a required field toplan/steps.jsonwithout updating every downstream reader. Ends onhandoffKind='deployable-app'for the happy path; second case verifies therejectladder rung still demotes through todesign-only. - Earlier in this changeset:
- diff-review GenUI \u2192
review/decision.jsonbridge (Phase 8 entry slice). Newapps/daemon/src/plugins/atoms/diff-review-genui-bridge.tsowns the__auto_diff_review_prefix detection, strict JSON validation of the surface payload (rejects non-object payloads + unknown decisions; coerces non-string entries out of the file lists; forwards optionalreason), and the end-to-endapplyDiffReviewDecisionToCwd({ cwd, value, reviewer })glue.POST /api/runs/:runId/genui/:surfaceId/respondnow bridges the choice surface response intorunDiffReview()so the user's decision lands on<cwd>/review/decision.jsonimmediately. Best-effort: failures are surfaced on the response payload asdiffReviewBridge: { ok: false, error }without regressing the GenUI respond contract. - Earlier in this changeset:
- Native diff-review UI on
GenUISurfaceRenderer(Phase 8 entry slice). NewDiffReviewChoiceSurfacecomponent renders the auto-derived__auto_diff_review_<stageId>choice surface natively: three top-level buttons (Accept all / Reject all / Partial…), an optional notes textarea forwarded asdecision.reason, and a per-file accept/reject checklist that appears when the user picks Partial. Local validation refuses Partial submission when any touched file is left undecided (mirrors the daemon'spartial must cover every touched filecontract). NewGenericChoiceSurfacerenders any schema with a primary enum property as a button group; property literally nameddecisionwins so plugin-author-customised diff-review schemas keep rendering as accept/reject/partial buttons. PendingSurface gains an optionalcontext: { touchedFiles?: [] }field so future runtime context (per-stage hints) can plug in without bloatingGenUISurfaceSpec. figma-extractasset rasterisation second pass (Phase 6 entry slice).runFigmaExtract({ offlineAssets: false })now downloads per-leaf-node assets via the FigmaGET /v1/imagesendpoint. New knobs:assetFormat('svg' | 'png' | 'jpg' | 'pdf', default svg),assetMaxBytes(default 5 MiB). 50-id chunks, per-id failure isolation (a single CDN hiccup doesn't lose the batch), every issue lands inmeta.unsupportedNodes[]withreason='asset-too-large' | 'no download URL returned' | 'download <status>' | 'image fetch error: …' | 'figma error: …'.atomDigestis recomputed after the asset pass settles so the digest reflects which assets succeeded.- Earlier in this changeset:
token-mapatom (Phase 6/7 entry slice). NewrunTokenMap({ cwd, source?, designSystem, strict? })writes<cwd>/token-map/{colors,typography,spacing,radius,shadow,unmatched,meta}.json. Match strategies (deterministic, first-win): exact value match → normalised hex (#abc → #aabbcc) → fuzzy name (strips--/ds-/odds-/theme-prefixes). Unmatched lands with'no-target-equivalent','target-collision', or'invalid-source'.parseDesignSystemTokens(body)heuristic helper lifts CSS custom properties + markdown table rows from a DESIGN.md body so daemon callers don't need a hand-curated bag.figma-extractatom (Phase 6 entry slice). NewrunFigmaExtract({ cwd, fileUrl?, fileKey?, token, fetchFn?, offlineAssets? })walks Figma's RESTGET /v1/files/<key>into the canonical<cwd>/figma/{tree,tokens,meta}.json. Token lift heuristics: SOLID fills + strokes → colours,cornerRadius→ radius, FRAME / GROUP heights → spacing candidates. Gradient + image fills land inmeta.unsupportedNodes[]with a reason.fetchFnis pluggable (tests inject stubs); offline mode keeps the assets/ directory empty. The OAuth bearer token is forwarded via theAuthorizationheader and never persisted by the runner — the daemon's connector-gate stays the only place that touches the token store.- Auto-derived
choicesurface fordiff-review(Phase 8 entry slice). Mirrors the connector-gate's autooauth-promptpattern. When a stage in the EFFECTIVE pipeline (consumer-declared OR scenario-fallback) listsdiff-review, applyPlugin() synthesises an implicitchoiceGenUI surface (__auto_diff_review_<stageId>,persist='run', 24h timeout,onTimeout='abort') so the user can accept / reject / partial without the plugin author having to declare the surface by hand. Plugin-author-declared surfaces with the same id win. - Earlier in this changeset:
- Bundled-scenario pipeline fallback (spec §23.3.3). New pure helper
resolveAppliedPipeline({ manifest, scenarios })returns{ pipeline, source: 'declared'|'scenario'|'none', scenarioId? }.applyPlugin()now consults the bundled scenarios surfaced byloadPluginRegistryView()and copies the matching scenario's pipeline verbatim into both theApplyResult.pipelineandAppliedPluginSnapshot.pipeline. Scenario plugins themselves never recurse.manifestSourceDigeststays unchanged across the fallback so spec §11.5 e2e-2 invariance holds. Only rows withsource_kind='bundled'ANDod.kind='scenario'are eligible — third-party scenarios cannot promote themselves. design-extractatom (Phase 6/7 entry slice). NewrunDesignExtract({ cwd, repoPath })readscode/index.json, walks every scannable file (css/scss/ts/tsx/js/jsx/html/json), and writes the canonical token bag at<cwd>/code/tokens.json. Captures hex / rgba / hsla colours, CSS custom properties (--*-color/-bg/-accent/-primary/-secondary/-surface/-border/-muted), font-family declarations, px/rem/em spacing on padding/margin/gap/inset, border-radius, box-shadow, and Tailwind config quoted hex palette entries. Each token recordssources[](<path>:<line>) +usage[]sotoken-maphas a precise audit trail.rewrite-planatom (Phase 7 entry slice). NewrunRewritePlan({ cwd, intent? })produces<cwd>/plan/{plan.md, ownership.json, steps.json, meta.json}. Heuristic ownership classifier maps every imported file to a tier (leaf | shared | route | shell); the step generator emits onerewrite-<slug>per leaf component, bundles sibling stylesheets, prepends atokens-alignmentstep when design-extract surfaced literals, marks shared/route touchesmediumrisk, and always closes with abuild-teststep.meta.atomDigestis over a canonicalised view ofcode/index.jsonso re-walks that don't change the file roster don't invalidate the plan.patch-editatom (Phase 7 entry slice). NewapplyPatchForStep({ cwd, stepId, diff }),skipStep(), andreadPlanProgress(). The applier parses unified-diff text (plain edits,/dev/nullcreation,/dev/nulldeletion, multi-file hunks) and enforces the spec §20.3 safety contract before any byte is written: (a) path-traversal guard, (b) every touched file MUST appear instep.files[], (c)shell-tier files requirestep.risk='high', (d) context lines must match exactly so stale patches surfacecontext mismatch at line Nwithout mutating disk. After a successful apply, the runner updatesplan/steps.json[id].status='completed'and writesplan/receipts/step-<id>.jsonwith files / added / removed / rationale.diff-reviewatom (Phase 7-8 entry slice). NewrunDiffReview({ cwd, decision? })readsplan/receipts/and emits<cwd>/review/{diff.patch, summary.md, decision.json, meta.json}. Decision composition rules:acceptdefaultsaccepted_filesto every touched file;rejectdefaultsrejected_filesto every touched file;partialMUST cover every touched file via theaccepted_files ∪ rejected_filesunion or the runner throwsmissing <file>so the GenUI surface can re-prompt.decision.jsonis round-trippable so a re-run without an explicit decision returns the persisted choice.- Earlier in this changeset:
build-testatom (Phase 7 entry slice). NewrunBuildTest({ cwd, buildCommand?, testCommand? })shells out to typecheck + test commands (overrides win; otherwise inferred frompackage.jsonscripts). Auto-detects pnpm / yarn / bun / npm from lockfile presence. Per-command timeout default 5 min, log budget 1 MiB. Emitsbuild.passing+tests.passingsignals (newly added to spec §22.4 vocabulary) plus the legacycritique.score.writeBuildTestReport()persistscritique/build-test.json+critique/build-test.log.code-importatom (Phase 7 entry slice). NewrunCodeImport({ repoPath, cwd })walks a real repo and writes a normalised<cwd>/code/index.jsonsnapshot. Honours a 60 s walk budget; skipsnode_modules,.git,dist,build,coverage, etc.; rejects symlinks. Detects framework (next / vite / remix / astro / sveltekit / cra / custom), package manager, style system, and routing model.handoffatom +ArtifactManifestprovenance fields (Phase 7-8 entry slice).ArtifactManifestgains the spec §11.5.1 reserved fields (sourcePluginSnapshotId,sourcePluginId,sourceTaskKind,parentArtifactId,artifactKind,renderKind,handoffKind,exportTargets[],deployTargets[]). NewrecordHandoff()helper enforces append-onlyexportTargets/deployTargetsand monotonichandoffKindpromotion (design-only<implementation-plan<patch<deployable-app).isDeployableAppEligible()centralises the §11.5.1 promotion rule (build.passing+tests.passing+ at least one docker / cli export target). Contracts barrelindex.tsswitched to.js-suffixed re-exports so daemon NodeNext resolution picks every type up end-to-end.- Bundled scenario plugins (spec §23.3.3). Four
od.kind: 'scenario'plugins underplugins/_official/scenarios/{od-new-generation,od-figma-migration,od-code-migration,od-tune-collab}/lock the default reference pipeline pertaskKind. Replacing a default for an enterprise / vertical edition is now a content edit rather than a daemon code change. The bundled boot walker registers them via the existing tier layout.
-
Plugin & marketplace system — earlier landing. Spec:
docs/plugins-spec.md. Living plan:docs/plans/plugins-implementation.md.OD_SNAPSHOT_RETENTION_DAYSreferenced-row TTL (PB2).pruneExpiredSnapshotsnow retires referenced snapshot rows whose project has been deleted AND whoseapplied_atis older than the configured window. Live-project rows stay pinned forever (reproducibility wins). The GC worker readsreadPluginEnvKnobs().snapshotRetentionDaysso the env-var contract spec PB2 reserved is now end-to-end.OD_BUNDLED_ATOM_PROMPTS=1activatescomposeDaemonSystemPrompt's atom-block path. When set AND the run carries an applied snapshot with a non-emptyod.pipeline.stages[*], the daemon walks each stage, callsloadAtomBodies+renderActiveStageBlock, and threads the result ascomposeSystemPrompt({ activeStageBlocks }). Default behaviour (flag unset) is byte-equal to today's prompt.- Phase 6 / 7 / 8 atom SKILL.md substrate. Nine new
plugins/_official/atoms/<atom>/{SKILL.md, open-design.json}pairs the bundled boot walker registers on startup:figma-extract,token-map(Phase 6);code-import,design-extract,rewrite-plan,patch-edit,diff-review,build-test(Phase 7);handoff(Phase 8). The fragments teach the agent what each (planned) atom expects so a plugin author who references one of these ids inod.pipeline.stages[*].atoms[]sees the canonical fragment without a doctor warning. The matching shell-out implementations stay scheduled per spec §16 Phase 6 / 7 / 8.
-
Plugin & marketplace system — earlier landing. Spec:
docs/plugins-spec.md. Living plan:docs/plans/plugins-implementation.md.- Per-cloud Helm value overrides.
tools/pack/helm/open-design/values-{aws,gcp,azure,aliyun,tencent,huawei,self}.yamlship the volume + ingress diffs spec §15.5 enumerates. Operators install withhelm install od ./tools/pack/helm/open-design -f values-aws.yaml. composeSystemPrompt({ activeStageBlocks }). Both daemon and contracts composers accept a pre-rendered list of## Active stageblocks (produced byrenderActiveStageBlock+loadAtomBodies). Substrate slice for the §23.3.2 prompt-fragment migration; the actual call-site wiring stays gated on the next phase so default behaviour is byte-equal to today's prompt.- Plugin-bundled component surface.
GenUISurfaceRenderermounts asandbox="allow-scripts"iframe at/api/plugins/:id/asset/<path>when a surface declaresod.genui.surfaces[].component. Communication is one-way viapostMessage({ kind: 'genui:respond', surfaceId, value }). The daemon-side asset endpoint serves files frominstalled_plugins.fs_pathunder the §9.2 preview CSP (default-src 'none'; connect-src 'none'; frame-ancestors 'self') plusX-Content-Type-Options: nosniff. ProjectStorage+DaemonDbadapter substrate. Newapps/daemon/src/storage/module ships the Phase 5 §15.6 interface contracts.LocalProjectStorage(v1 default) is fully wired and tested;S3ProjectStorageis an interface-locked stub that throws on every op until the AWS SDK wiring lands.resolveDaemonDbConfig({})parsesOD_DAEMON_DB/OD_PG_*env vars but the SQLite path remains the only reachable backend in v1.
- Per-cloud Helm value overrides.
-
Plugin & marketplace system — earlier landing. Spec:
docs/plugins-spec.md. Living plan:docs/plans/plugins-implementation.md.- Phase 5 bound-API-token guard.
startServer()refuses to bind a non-loopbackOD_BIND_HOSTwithoutOD_API_TOKEN; bearer middleware on/api/*rejects non-loopback peers withoutAuthorization: Bearer <OD_API_TOKEN>./api/health,/api/version,/api/daemon/statusstay open so monitoring probes (kubelet, Compose) work without secrets. - Helm chart templates.
tools/pack/helm/open-design/templates/ships Deployment, Service, Secret, ConfigMap, two PVCs, optional Ingress, plus _helpers.tpl + NOTES.txt. The chart installs end-to-end withhelm install od ./tools/pack/helm/open-design --set secrets.apiToken=$(openssl rand -hex 32). od.genui.surfaces[].component.GenUISurfaceSpecSchemaaccepts a{ path, export?, sandbox? }field;genui:custom-componentjoinsKNOWN_TOP_LEVEL_CAPABILITIES;doctorPlugin()flags the missing-capability + path-traversal cases. The component path is the v1 substrate for spec §10.3.5 alignment-roadmap row 2; the web sandbox loader stays scheduled..github/workflows/docker-image.yml. Multi-arch (linux/amd64 + linux/arm64) build + push to ghcr.io::edgeon main,:<version>+:lateston tag,:sha-<short>on every push, smoke build on PRs. Authenticates via GITHUB_TOKEN withpackages:write.
- Phase 5 bound-API-token guard.
-
Plugin & marketplace system — earlier landing. Spec:
docs/plugins-spec.md. Living plan:docs/plans/plugins-implementation.md.@open-design/agui-adapterworkspace package +GET /api/runs/:runId/agui. Pure-TS bidirectional bridge between OD's nativePersistedAgentEvent/GenUIEvent/PluginPipelineStageEventunion and the AG-UI canonical event protocol. The new SSE endpoint mirrors/api/runs/:id/eventsbut pipes every record throughencodeOdEventForAguiso a CopilotKit / AG-UI client consumes an OD run unmodified. v1 plugins need no change to be consumable inside the AG-UI ecosystem (spec §10.3.5).renderActiveStageBlock+loadAtomBodies. Substrate slice for spec §23.3.2 patch 2: the daemon-side helper reads<bundled-fsPath>/SKILL.mdfor any registered bundled atom and the contracts-side renderer assembles a## Active stage: <id>block. ThecomposeSystemPrompt()rewiring that consumes them is the next PR; today the helpers are reachable, tested, and the bundled atom plugins from §3.I3 already ship the matching SKILL.md bodies.- Phase 5 Dockerfile + docker-compose + Helm chart entry slice.
deploy/Dockerfilenow bundlesplugins/_official/soregisterBundledPlugins()finds the atom set inside the container.tools/pack/docker-compose.ymlis the canonical hosted-mode manifest (two-volume layout, OD_API_TOKEN, /api/daemon/status healthcheck).tools/pack/helm/open-design/pins the Helm chart parameter surface for the per-cloud value overrides spec §15.5 enumerates; templates land in the Phase 5 follow-up PR.
-
Plugin & marketplace system — earlier landing. Spec:
docs/plugins-spec.md. Living plan:docs/plans/plugins-implementation.md.- Pipeline runner wired into
POST /api/runs. Plugin runs whose snapshot carriesod.pipeline.stages[*]now emitpipeline_stage_startedsynchronously before the agent process spawns. Subsequent stage events (pipeline_stage_completed, per-stagerun_devloop_iterationsaudit rows, devloop convergence) fire asynchronously while the agent runs. Stage runner is a converging stub (critique.score=4,preview.ok=true) so single-pass pipelines walk through every stage in O(stages) time; loop stages still respectOD_MAX_DEVLOOP_ITERATIONS. Errors surface aspipeline_stage_failedevents and never block the agent. e2e-3 flips from entry-slice to the full §8 contract:apps/daemon/tests/plugins-headless-run.test.tsasserts the first SSE event on a pipeline-bearing plugin run ispipeline_stage_started. od doctor. Repo-wide diagnostics: daemon status, installed-plugin doctor sweep, library inventory (skills / design-systems / atoms / craft). Exits 1 when any plugin doctor returns ok=false; exit 64 when the daemon is unreachable.od config get/set/list/unset. WrapsGET/PUT /api/app-config. Top-level keys via positional or--value; nested values via--value-json.- Phase 4 / §23 entry slice — bundled atom plugins. New
plugins/_official/atoms/{discovery-question-form,todo-write,direction-picker,critique-theater}/{SKILL.md,open-design.json}pairs, plus a daemon boot walker (apps/daemon/src/plugins/bundled.ts) that registers each folder undersource_kind='bundled'/trust='bundled'on every startup. Idempotent (upserts), ENOENT-silent (works outside the dev tree). Lays the substrate for the §23.3.2 patch that lifts prompt fragments out ofsystem.ts.
- Pipeline runner wired into
-
Plugin & marketplace system — Phase 2A + 1 + 1.5 + 2B + 2C entry slice + 3 (full) + 4 (scaffold / export / publish / atoms doc / library CLI) + early 5 (earlier landing). Spec:
docs/plugins-spec.md. Living plan:docs/plans/plugins-implementation.md.- Phase 4 publish.
od plugin publish <id> --to anthropics-skills|awesome-agent-skills|clawhub|skills-sh [--repo <github>] [--open]builds the catalog submission URL + PR body and (with--open) auto-launches the system browser. The author still goes through the upstream review flow; OD never POSTs anywhere. - CLI parity remainder.
od atoms list/show,od skills list/show,od design-systems list/show,od craft list/show,od status,od version. New HTTP routesGET /api/craft+GET /api/craft/:idwalk thecraft/directory and return{ id, label, bytes }summaries. od marketplace search "<query>" [--tag <tag>]. Substring match over every configured marketplace'splugins[]; powered entirely by the catalog metadataod marketplace addalready cached, so a code agent can discover plugins without being inside the desktop UI.
- Phase 4 publish.
-
Plugin & marketplace system — Phase 2A + 1 + 1.5 + 2B (composer mount + marketplace deep UI) + 2C entry slice + 3 entry slice + 4 (scaffold / export / atoms doc) + early 5 (earlier landing). Spec:
docs/plugins-spec.md. Living plan:docs/plans/plugins-implementation.md.- Phase 4 author tooling.
od plugin scaffold --id <id>writes a starter SKILL.md + open-design.json + README.md (optionally a.claude-plugin/plugin.json).od plugin export <projectId> --as od|claude-plugin|agent-skill --out <dir>materialises a publish-ready folder from the AppliedPluginSnapshot behind a project (or a snapshot id), so any chat that ran a plugin can be re-published to anthropics/skills, awesome-agent-skills, clawhub, or skills.sh without leaving the terminal. The output's open-design.json carries a provenance block (snapshotId + manifestSourceDigest + appliedAt) that reverse-resolves to the originating run. - Phase 2B marketplace deep UI. New routes
/marketplaceand/marketplace/<pluginId>rendered byMarketplaceView(catalog grid + trust filters + configured catalogs panel) andPluginDetailView(manifest, capability checklist, connector requirements, declared GenUI surfaces, "Use this plugin" → applyPlugin → Home)./plugins/<id>is a parsed alias so the public-site URL scheme reserved in spec §13 already works in-app. - Phase 2B ChatComposer mount.
ChatComposerrenders<PluginsSection variant="strip" />above the input whenever aprojectIdis known. Apply hydrates the draft only when empty so a mid-typing user is never overwritten. docs/atoms.md. New canonical reference for the first-party atom catalog: implemented vs planned ids, task-kind mapping, how the daemon resolves an atom at run time, the closeduntil-signal vocabulary, and the §22.5 community-plugin → first-party-atom promotion path.
- Phase 4 author tooling.
-
Plugin & marketplace system — Phase 2A finished + Phase 1 follow-up + Phase 1.5 + Phase 2B (composer mount) + Phase 2C entry slice + Phase 3 entry slice + early Phase 5 (earlier landing). Spec:
docs/plugins-spec.md. Living plan:docs/plans/plugins-implementation.md.- CLI canonical agent-facing API (spec §11.7). Every UI action now has a CLI equivalent:
od project create/list/info/delete,od run start/watch/cancel/list/info(with--plugin,--inputs,--grant-caps,--snapshot-id,--follow),od files list/read/write/upload/delete,od conversation list/info. Plugin runs route through the §3.A1 snapshot resolver. - Phase 1.5 headless lifecycle. New
od daemon start [--headless] [--serve-web] [--port] [--host],od daemon status [--json],od daemon stop. Backed by new HTTP routesGET /api/daemon/statusandPOST /api/daemon/shutdown(loopback-only). The defaultod(no subcommand) keeps its desktop behaviour for back-compat. e2e-3's headless install→project→run loop is now anchored in a daemon supertest (apps/daemon/tests/plugins-headless-run.test.ts). - Phase 3 marketplace plugin install resolution.
POST /api/plugins/installwith a bare plugin name walks every configuredplugin_marketplacesrow in registration order and re-routes to the canonicalgithub:…/https://…source recorded in the matched manifest. CLI:od plugin install <name>works against any catalog the operator added viaod marketplace add <url>. - Web composer mount. New
PluginsSectionwidget combinesInlinePluginsRail,ContextChipStrip,PluginInputsForm, plusrenderPluginBriefTemplate's{{var}}substitution. Mounted inNewProjectPanelbelow the project-name input as the Phase 2A discovery surface. Purely additive: the existing Send button rules are unchanged. - The earlier work in this changelog block remains in place (snapshot resolver, trust mutation, connector tool-token gate, fallback rejection, GitHub tarball + HTTPS install sources, pipeline runner with cross-conversation cache, marketplace registry, snapshot GC worker, web component primitives, definition-of-done suite, …).
- CLI canonical agent-facing API (spec §11.7). Every UI action now has a CLI equivalent:
-
Plugin & marketplace system — Phase 2A → entry slice of Phase 2B/2C/3 (earlier landing). Spec:
docs/plugins-spec.md. Living plan:docs/plans/plugins-implementation.md.- Snapshot resolver wires
applyPlugin()intoPOST /api/projectsandPOST /api/runs. Capability gate failures map to HTTP 409 / exit 66; missing inputs map to HTTP 422 / exit 67. The in-memory run object carriesappliedPluginSnapshotIdso connector / replay paths read a frozen view. - Trust mutation: new
POST /api/plugins/:id/trustendpoint plus matchingod plugin trust <id> --capabilities …CLI. Validates the spec §5.3 capability vocabulary (rejects unknown / malformed strings); preserves the implicitprompt:injectfloor on revoke. - Connector tool-token gate: tokens minted for plugin runs carry
pluginSnapshotId/pluginTrust/pluginCapabilitiesGranted./api/tools/connectors/executere-validates the §5.3connector:<id>rule per call (CONNECTOR_NOT_GRANTED 403 on miss). Trusted plugins implicitly carryconnector:*; restricted plugins must list each id explicitly. - API-fallback rejection: every
/api/proxy/*entry returns 409 PLUGIN_REQUIRES_DAEMON when a body smugglespluginId/appliedPluginSnapshotId. - Snapshot GC worker enforces the PB2
expires_atTTL (OD_SNAPSHOT_UNREFERENCED_TTL_DAYS/OD_SNAPSHOT_GC_INTERVAL_MS) at boot + on a periodic interval. Operator escape hatch:od plugin snapshots prune --before <ts>. - Installer accepts
github:owner/repo[@ref][/subpath](codeload tarball) andhttps://*.tar.gzarchives in addition to local folders. Hard guards: symlink / hard-link rejection, path-traversal segments, 50 MiB size cap. - Pipeline runner emits
pipeline_stage_*events per stage and persists every iteration intorun_devloop_iterations. Pre-stage GenUI surfaces auto-derived for not-yet-connected required connectors hit the cross-conversation cache (e2e-5: a second conversation in the same project never re-broadcasts an already-resolvedoauth-prompt). - Marketplace registry minimum verbs:
od marketplace add/list/info/refresh/remove/trustplus the matching HTTP routes. Default trust tier isrestrictedper spec §9; the Phase 3 follow-up wiresod plugin install <name>resolution + the trust UI. - Web composer surface:
applyPlugin()state helper,InlinePluginsRail,ContextChipStrip,PluginInputsForm,GenUISurfaceRenderer(confirmation + oauth-prompt first-class; form / choice fall back to a JSON Schema preview until Phase 2A.5),GenUIInboxdrawer. od plugin runapply→start shorthand. CLI structured error helper maps recoverable HTTP 4xx envelopes to the §12.4 exit codes (64–73) so code agents get a stable retry contract.- Plan §8 e2es covered at the daemon level: e2e-2 (pure apply across runs), e2e-4 (replay invariance after plugin upgrade via
renderPluginBlock(snapshot)), e2e-5 (GenUI cross-conversation cache), e2e-6 (connector trust gate; re-validated independently in the token-issuance + execute routes), e2e-7 (API-fallback rejection at every proxy entry), e2e-8 (apply purity regression: 100 applies → 0 FS mutation, +100 snapshot rows). e2e-3 (headless run) stays scheduled for Phase 1.5.
- Snapshot resolver wires
-
ib-pitch-bookskill — investment-banking strategic-alternatives pitch book (Anthropic financial-services Pitch Agent workflow); shipsexample.htmland IB layout references.
0.6.0 - 2026-05-09
A connectivity-and-iteration release: Open Design becomes a fully bidirectional MCP citizen (external MCP client with 39 templates), ships Cloudflare Pages deployment for generated artifacts (with custom domains), advances Critique Theater to Phase 6 (interrupt + project-keyed run registry), and lands a redesigned top bar, draggable file tabs, batch delete, vector PDF export, agent-callable research/search, and Orbit activity summaries. Hyperframes learns the HTML-in-Canvas API. New BYOK provider (Ollama Cloud), new agent capabilities (Gemini 3 preview + GPT-5.1 codex picker + DeepSeek v4), new design systems (BMW M, Slack, Cisco, Webex, Mission Control, Urdu Modern), eight new skill bundles, and Turkish + Thai locales. 136 merged PRs since 0.5.0.
- External MCP client with daemon-managed OAuth and 39 design-focused templates. Open Design can now consume MCP servers, not just expose itself as one. (#898)
- Cloudflare Pages artifact deployment. One-shot publish of generated artifacts to Pages from the desktop app. (#729)
- Cloudflare Pages custom domains. Bind your own domain to deployed artifacts. (#851)
- Preserve OAuth state and advertised tool counts when reconnecting MCP/connector providers. (#1036)
- Optimized Composio connector previews. (#907)
- Phase 6.1: critique interrupt endpoint + project-keyed run registry. Long critiques can now be interrupted cleanly per project. (#819)
- Shared
CritiqueRoundSummary/CritiqueRunStatustypes via the@open-design/contractspackage. (#1016)
- Top bar redesign — Share/Present lifted to the top bar, zoom dropdown, and an explicit focus toggle. (#1048)
- Draggable file tab reordering in the workspace. (#936)
- Batch delete for selected design files. ([#783])
- Sortable Design Files table columns. (#804)
- Privacy consent choices made explicit at first launch. (#1031)
- Differentiated "recent" vs "your designs" sorting. (#845)
- Inspect / Picker now renders an empty-annotation state instead of a blank panel. (#1005)
- Toggle to reveal saved media-provider API keys. (#867)
- Direct PDF export for artifacts. (#532)
- Hyperframes skill learns the HTML-in-Canvas API for richer in-canvas previews. (#852)
- Consolidated Hyperframes video template updates. (#1079)
- Inspect overlay support on Windows packaged builds. (#944)
- Allow
od://URLs throughsetWindowOpenHandlerso live-artifact previews open in a child window. (#933)
- Import existing local folder as a project. (#624)
- Agent-callable research command +
/search. Agents can ask the project for grounded research without leaving the chat. (#615) - Orbit activity summaries. (#681)
- Finalized the design-package endpoint (closes #450). (#832)
- Closed pi adapter parity gaps (
imagePaths,extraAllowedDirs, error events,sendAgentEventrouting). (#763) - Language-boost support for Minimax TTS. (#773)
- Expose Gemini 3 preview models and Gemini 2.5 Flash Lite in the picker. (#986)
- Add GPT-5.1 entries to the Codex picker. (#946)
- Expand Codex picker coverage. (#757)
- Stable nightly promotion gate for
[codex]. (#962) VP_HOMEenvironment variable support in agent resolution. (#859)- Auto-rebuild
better-sqlite3on Node.js ABI mismatch postinstall. (#813) - Increase agent inactivity timeout. (#1071)
- Reset inactivity watchdog on raw stdout bytes, not just parsed events. (#976)
- Ollama Cloud as a BYOK provider. (#923)
- Opt-in Langfuse telemetry. (#800)
- Make Azure API version optional. (#941)
ib-pitch-bookskill — investment-banking strategic-alternatives pitch book (Anthropic financial-services Pitch Agent port). (#888)github-dashboardskill. (#666)clinical-case-reportskill. (#581)social-media-matrix-trackerskill — live-artifact tracker. (#810)trading-analysislive-artifact dashboard skill. (#824)otd-operations-brieflive-artifact template. (#794)- 32 zhangzara HTML deck templates. (#704)
- 7 example dashboards + contract demo for the live-artifact skill. (#716)
after-hours-editorialtemplate skill. (#1053)swiss-user-research-videotemplate skill. (#1054)editorial-burgundy-principlestemplate skill. (#1065)swiss-creative-modetemplate skill. (#1068)- BMW M design system. (#579)
- Slack design system. (#899)
- Cisco and Webex design systems. (#991)
- Mission Control design system. (#858)
- Urdu Modern (Indus Script) design system. (#714)
- Craft
laws-of-uxmodule so generated UIs respect working-memory limits. (#809) - Craft
typography-hierarchyandtypography-hierarchy-editorialrules. (#975, #979)
- Turkish README translation. (#843)
- Full Thai (
th) UI locale. (#1018) - Renamed live-artifact tab label in zh-CN and zh-TW. (#969)
- Default
idlocale to English for keys not yet translated. (#822) - Trim BYOK proxy fallback line from zh-CN intro. (#915)
- Docker Compose deployment workflow. (#65)
- Preserve beta e2e spec reports in R2. (#812)
- Document the Colima build-swap helper. (#967)
- Vaunt contributor recognition (5-tier system). (#908)
- Hardened security scan findings and upgraded dependencies. (#806)
- Strengthened e2e PR coverage and entry/settings automation coverage. (#796, #811)
- Refreshed contributors wall and GitHub metrics. (#856, #1004, #853, #998)
- Refined
typography-hierarchycraft docs — clarify edge cases and make lint measurable. (#979)
- MCP install snippet survives daemon port changes. (#846)
- Pin
OD_DATA_DIRin/api/mcp/install-infoenv so the macOS-packaged MCP server stops EPERM'ing on.od/projects. (#857) - Reserve clearance for the MCP server Copy button so it stops overlapping the snippet. (#847)
- Give the MCP server Copy button a solid surface so it reads against the code block. (#840)
- Stable curated tool count in the connector card badge. (#767)
- Remove redundant "Connect GitHub" placeholder from the import menu. (#964)
- Connector "Close window" button always gives feedback. (#995)
- Confirm before clearing the saved Composio API key. (#877)
- Keep saved Composio API key indicator visible while typing a replacement. (#751)
- Confirm before clearing a saved Media provider API key. (#875)
- Cloudflare Pages custom-domain lookup. (#958)
- Surface explicit error/retry state when example preview HTML fails to load. (#863)
- Confirm before closing a dirty sketch so unsaved strokes are not lost. (#988)
- Keep chat auto-scroll glued to the bottom across streaming chunks. (#989)
- Preserve Chat scroll position across Chat/Comments tab switches. (#841, #886)
- Differentiate selected, hover, and focus states in the language switcher. (#987)
- Scroll the active workspace tab into view when the strip overflows. (#990)
- Keep the Design Files tab visible when workspace tabs scroll. (#842)
- Wrap long note text inside picker/comment popovers. (#830)
- Wrap comment-popover action row so the Save/Sending button can't exceed the popover edge. (#829)
- Prevent comment popover header overflow when the label is too long. (#833)
- Truncate long Inspect-panel labels so they cannot spill past the panel edge. (#838)
- Keep Inspect-panel close button on a stable single-line layout. (#839)
- Increase project meta line-height to prevent descender clipping. (#834)
- Give the deploy modal primary action more breathing room. (#992)
- Hide the unsupported "Save comment" button on Pods selections. (#993)
- Clear stale upload error banner when previewing existing files. (#994)
- Expand design file row click target. (#1039)
- Keep entry footer pills compact. (#1045)
- Hide stale upload error banner when previewing other files. (#994)
- Scope settings save validation + sanitize payload to the active sidebar section. (#827)
- Ensure the Settings close button is always clickable. (#971)
- Correct
srcdocinjection and deck bridge for JS strings containing closing</script>. (#938) - Unbreak the Create button on plain HTTP / LAN-IP deployments. (#900)
- Differentiate recent vs your-designs sorting. (#845)
- Keep examples filter counts consistent. (#949)
- Cleanly quit the macOS packaged app. (#422)
- Keep modal controls clickable in drag regions. (#1032)
- Improve Orbit and packaged data-dir startup errors. (#1067)
- Fix desktop preview interactions and connector auth feedback. (#864)
- Fix desktop preview and packaged app interactions. (#879)
- Fix desktop prompt template close hitbox. (#1056)
- Pack/win: close detection gaps that let
Open Design.exestay locked at install time. (#823) - Tools-pack: mark
blake3-wasmas external in the macOS prebundle. (#844) - Packaged: swallow harmless
setTypeOfService EINVALfrom undici. (#906)
- Settle completed runs and clean up shutdown children. (#924)
- Fix stuck chat runs and unintended cancels. (#896)
- Write SSE events atomically in
createSseResponse.send. (#972) - Media generation task state survives daemon restart (#648). (#884)
- Sync Orbit last run with the selected prompt template. (#937)
- Image template creations execute the selected prompt automatically. (#752)
- Serve Python files as text. (#947)
- Type-check core server paths and leaf modules. (#943, #952)
- OpenCode todowrite footer state. (#1046)
- Stale internal links across docs. (#950)
- Repository-wide code review guidelines. (#927)
- Design system authoring guide. (#961)
- Skills contributing guide. (#1035)
- Docker setup instructions in QUICKSTART, CONTRIBUTING, and README. (#935)
- Re-add
awesome-design-mdreference to QUICKSTART. (#940) - Update prompts path from web to daemon in README files. (#756)
- Test: cover model option rendering. (#948)
- Test: de-flake chat-scroll-preservation across tab switches. (#886)
- Auto-generated metrics + contributors wall refreshes. (#853, #998, #856, #1004)
- Release: Open Design 0.5.0 changelog landing. (#820)
0.5.0 - 2026-05-07
A minor release focused on iteration: live-data dashboards graduate to a first-class artifact category, an in-preview Inspect mode lands for per-element style tuning, the desktop launcher gets an accent color theme, Critique Theater advances to Phase 5, and Linux gains headless lifecycle support. New Qoder CLI agent, Nano Banana image provider, and Indonesian locale. 51 merged PRs since 0.4.1, accumulated across 16 beta cycles.
- Inspect mode — live per-element style tuning in the HTML preview. (#362)
- Accent color control + launcher — a global accent persists across the desktop launcher and entry view. (#683)
- Connection tests for execution settings — verify provider config without launching a chat. (#507)
- Replaced the SketchEditor
window.prompt()text tool with an in-app modal so long prompts stop getting clipped. (#738)
live-dashboardskill — generic Live Artifact dashboard template. (#778)clinic-consolelive-artifact template. (#795)- FlowAI live dashboard template skill. (#801)
- Notion-style team dashboard prompt template (Live Artifact). (#799)
waitlist-pageskill. (#555)social-media-dashboardskill + Totality Festival design system. (#678)- Five Orbit briefing prompt templates. (#671)
- Craft
form-validationmodule — generated forms follow modern RHF/Zod patterns instead of 2018 Formik habits. (#625)
- Phase 5 — panel prompt template + system composer wiring. (#524)
- Qoder CLI agent adapter. (#626)
- Project transcript export to disk for downstream tools (replay, audit, sharing) — prereq for #450. (#493)
- Override the Codex executable path for nvm / mise / fnm-installed toolchains. (#755)
- Codex image projects can use built-in imagegen. (#622)
- DeepSeek v4 models in the model catalog. (#722)
OD_LEGACY_DATA_DIRmigrator for 0.3.x → 0.4.x data recovery. (#712)
- Nano Banana image provider. (#631)
- HyperFrames video previews, provider badge, and source filter on the templates surface. (#293)
- Linux headless lifecycle —
install/start/stopfrom CLI without a desktop session. (#686) - Improved Windows beta packaging and installer flow. (#768)
- Migrated beta release publishing to R2. (#805)
- Indonesian (
id) UI locale. (#414)
- Project file watcher now ignores
.venvand other large dirs so Python projects stop overwhelming it. (#531) - Daemon CORS whitelist accepts portless
Originheaders for Chrome compatibility. (#735) - Extended OpenAI image request timeouts so larger generations stop being killed mid-flight. (#788)
- Surfaced the
@nexudotioX account in README and entry sidebar. (#696)
- Delivered Copilot prompts via stdin to avoid Windows
ENAMETOOLONG. (#727) - Surfaced OpenCode error frames; treated empty-output runs as failed instead of silently succeeding. (#700)
- Discovered toolchain paths for GUI-launched agents on minimal
PATH. (#614)
- Removed Tweaks-mode element-selector tooltip noise. (#697)
- Fixed chat pane overflow. (#740)
- Narrowed the
ws-tabs-barscrollbar so filenames stop overlapping. (#781) - Improved settings dialog scroll behavior. (#667)
- Widened settings subtitle so the English copy fits on one line. (#747)
- Persisted design system selection across sessions. (#621)
- Aligned the design system default test fixture. (#708)
- Showed an alert when the PDF export popup is blocked. (#664)
- Fixed the Windows link-code-folder dialog. (#698)
- Made desktop entry chrome consistent. (#655)
- Unbroke Claude Design ZIP import on Node 24 and raised the file ceiling. (#591)
- Diagnosed missing Next package during
tools-devweb startup. (#675)
- Aligned
README.esUI references to thees-ES.tslocale. (#611) - Fixed Ukrainian prompt template translations and removed duplicate keys. (#674, #680)
- Documented the Linux namespace env var in
tools-pack. (#670) - Fixed broken
pi-ailinks after the package split. (#277)
- Added desktop settings + project flow e2e coverage. (#306)
- CI: notify Discord
#resolvedwhen issues are closed by a merged PR. (#685) - Refreshed generated GitHub metrics SVG and contributors wall. (#718, #720)
0.4.1 - 2026-05-06
0.4.1 is the startup hotfix for the broken 0.4.0 desktop packages. It restores packaged app startup on macOS and Windows, adds release validation so the failure mode is caught before publication, and includes the small UI, agent, documentation, i18n, and craft updates that landed while the hotfix was being verified.
- Manual edit mode for direct artifact edits. (#620)
- Cmd/Ctrl+P quick file switcher for faster project navigation. (#556)
- Resizable chat panel. (#563)
- Added model name to PI initial status and RPC abort on cancel. (#618)
- Craft
accessibility-baselinemodule with opt-ins for dashboard, HR onboarding, and mobile onboarding. (#587) - Craft
rtl-and-bidimodule so artifacts handle Arabic, Hebrew, and Persian content more reliably. (#595) - Added i18n structure checks. (#608)
- Updated README first-PR links so
help-wantedissues are surfaced alongsidegood-first-issue. (#605)
- Fixed packaged desktop startup by building
@open-design/contractstodist/*.mjs+.d.ts, pointing its exports at compiled JavaScript, and building contracts before all packaged lanes pack workspace tarballs. (#577) - Added packaged runtime beta gating so release candidates install, start, inspect
/api/health, collect logs, stop, and uninstall before promotion. (#637)
- Added the required stdio MCP server env field and recover from
-32602onsession/set_model. (#627) - Normalized ACP
mcpServersto the stdio shape for Kimi/Hermes ACP. (#612) - Fixed agent CLI configuration and workspace focus mode. (#604)
- Preserved error messages across conversation reloads. (#623)
- Kept chat recoverable after conversation load failures. (#637)
- Honored native macOS quit behavior in the packaged desktop shell. (#637)
- Documented
OD_DATA_DIRand migration from.od/to the Desktop app. (#570) - Added Chinese (Simplified) QUICKSTART. (#578)
- Backported missing zh-TW README sections from the English README. (#586)
- Synced and improved the Korean README. (#619)
- Refined release workflows, CI scope, e2e layout, and packaged runtime smoke coverage for beta validation. (#637)
- Refreshed generated GitHub metrics. (#592)
0.4.0 - 2026-05-05
A multi-protocol leap: Open Design now ships as an MCP server, ships Critique Theater (Design Jury) Phase 4, gains live-reload + Tweaks mode + live artifacts in the preview pane, and adds five new agent / runtime adapters. 71 merged PRs from 40+ contributors over two days. Linux AppImage packaging landed in tooling, but the stable Linux artifact is deferred from 0.4.0 while containerized release packaging is hardened.
od mcp— expose Open Design as a stdio MCP server. Coding agents in other repos (Claude Code, Codex, Cursor, VS Code, Antigravity, Zed, Windsurf) can read files from local Open Design projects directly, including the project the user has open in the Open Design app right now. (#399)- Link code folder support for agent context — point agents at any local code folder alongside the design project. (#455)
- Kilo CLI (ACP) agent adapter. (#480)
- DeepSeek TUI agent adapter. (#439)
- Critique Theater Phase 4 — persistence, transcript, and orchestrator. The "Design Jury" multi-panelist scoring pipeline is now end-to-end. (#481)
- Critique Theater foundation — shared contracts and streaming v1 parser (Phases 0–2). (#387)
- Live-reload preview iframes when project files change on disk. (#409)
- Tweaks mode for HTML previews — element picker, pod selection, batched chat attachments. (#513)
- URL-load HTML preview iframes by default (
?forceInline=1opt-out). (#384) - Live artifacts and Composio connector catalog. (#381)
- Linux x64 AppImage tooling in
tools-pack; stable release artifact deferred from 0.4.0 while the containerized packaging lane is hardened. (#369) - Optimize packaged mac artifact size. (#424)
OD_MEDIA_CONFIG_DIRto relocatemedia-config.json(Nix store, immutable images, sandboxes). (#411)- Modernized multi-provider API proxy routing (Anthropic, OpenAI-compatible, Azure OpenAI, Google Gemini). (#385)
- Seed daemon with pre-baked decks and web prototypes. (#457)
- Atelier Zero editorial collage landing-page design system. (#366)
open-design-landingrename, kami skill bundle, and landing OG assets. (#428)- Craft
animation-disciplinemodule + opt-ins on mobile-app, mobile-onboarding, gamified-app. (#515) - Craft
state-coveragemodule + opt-ins on dashboard, mobile-app, kanban-board. (#502)
- Skills & design systems management page in Settings. (#535)
- Batch ZIP download with multi-select. (#405)
- Complete French localization, README, and Quickstart. (#326, #397, #434)
- Ukrainian UI localization. (#395)
- Russian UI locale refresh + README + gallery metadata. (#393, #396)
- Brazilian Portuguese README translation. (#460)
- Arabic README translation. (#458)
- Bind daemon to localhost by default + origin validation. (#365)
- Strip
ANTHROPIC_API_KEYwhen spawning Claude Code. (#400) - Preserve
ANTHROPIC_API_KEYwhenANTHROPIC_BASE_URLis set. (#514) - Preserve
*_API_KEYenv vars for CLI agents in packaged builds. (#404) - Normalize daemon proxy origins. (#392)
- Resolve daemon
package.jsonfrom any compiled layout so the packaged app reports the correct version. (#537) - Correct Claude Code
--add-dircapability detection. (#440) - Handle ACP
-32603errors gracefully insession/set_model. (#492) - Expose skill resources via cwd-relative aliases. (#435)
- Support nested paths in project file serve route. (#401)
- Respect baseUrl path verbatim in OpenAI-compat proxy. (#410)
- Prevent vertical scrollbar on artifact preview frame. (#453)
- Prevent vertical scrollbar on
ws-tabs-bar. (#448) - Language option button height truncation in Settings. (#447)
- Aspect-ratio cards no longer overflow into siblings. (#476)
- Add copy buttons for FileViewer code blocks. (#471)
- Lowercase
todowritecompatibility in ToolCard. (#523) - Cap
htmlPreviewSlideStateMap to prevent memory leak. (#488) - Isolate preview blob export paths. (#429)
- Split execution-mode tabs and align active chip visuals. (#418)
- Tighten entry-tab layout and design-system showcase color picker. (#412)
- Lift coming-soon tip above sticky tabs and make it readable in dark theme. (#382)
- Fix file tab wheel scrolling. (#549)
- Clear selection on project switch. (#465)
- Copilot prompt processing with correct command format. (#466)
- Codex Gemini CLI trust handling. (#352)
- Show window on macOS dock activate. (#270)
- Bundle prompt templates in packaged desktop resources. (#417)
- Deploy with
npm wrangler. (#421)
- Discord invite badge in README. (#504)
- Surface desktop downloads in README. (#522)
- "Running the Project" section in README. (#468)
- First-PR link points to /contribute page. (#494)
- Defer README template-driven generation; capture #195 discussion. (#403)
- Fix typo in zh-TW README. (#548)
- Auto-generated metrics SVG and contributors wall refresh. (#406, #407, #489, #490)
- Enforce test directory conventions. (#496)
0.3.0 - 2026-05-03
A fast follow-up to 0.2.0 focused on richer design workflows, packaged-agent reliability, export/deploy flows, and broader internationalization. 39 merged PRs from 25 contributors.
- Pet companion with Codex hatch-pet integration. (#296)
- Brand design-system cards, thumbnails, and DESIGN.md side-by-side preview. (#289)
- Per-tool renderer registry for generative UI. (#282)
- Task completion sound and browser notification. (#359)
- Persist code-agent startup state. (#255)
- Mistral Vibe CLI agent adapter. (#354)
- Devin for Terminal support. (#301)
OD_BIND_HOSTand--hostfor interface binding. (#328)
- Taste-skill-derived web prototype and HTML PPT examples. (#358)
pptx-html-fidelity-auditskill wired into export prompts. (#307)- Broader PPTX fidelity script coverage beyond CJK. (#308)
- Native desktop Save As dialog for
.pptxdownloads. (#330) - Export as Markdown from the share menu. (#345)
/api/projects/:id/deploy/preflightfor pre-upload inspection. (#320)
- Include
nvm/fnm/miseagent CLI bins in packaged PATH. (#364) - Detect Codex and Gemini CLIs from user toolchain paths. (#346)
- Upgrade
better-sqlite3for Node 24 Windows prebuilt support. (#357) - Lead Copilot spawn with
-p -so prompt-via-stdin is consumed. (#351) - Drop literal
-argv from Codex spawn so prompts deliver via stdin pipe alone. (#342) - Wrap
cmd.exeshim invocations to survive/s /cquote stripping. (#339)
- Download as
.zipnow returns the actual project tree. (#341) - Keep Design Files view active after deleting a file. (#329)
- Scroll workspace tabs in place instead of the window. (#363)
- Treat inlined script content as literal in FileViewer. (#343)
- Use response-order matching for bulk upload aggregation. (#323)
- Serve
.jsx/.tsxwith JS-family MIME types so browser loaders accept them. (#340) - Fix macOS entry view drag region. (#373)
- Increase project upload limit from 20MB to 200MB. (#319)
- Bundle and rewrite assets referenced from inline
<style>blocks andstyle=""attributes. (#314)
- Update locale coverage after main merge. (#251)
- Add missing
designFiles.showMorekeys toar,hu,ko,pl, andtr. (#335)
- Japanese documentation update. (#309)
- README contributors wall refresh. (#360)
- Spelling fixes in CLI comments, spec, and video prompt docs. (#300)
0.2.0 - 2026-05-02
A feature-heavy follow-up to 0.1.0 — dark mode, xAI Grok Imagine media generation, headless deploy mode, OpenClaude fallback, four new locales, and a much richer skill / design-system / prompt-template catalog. 45 merged PRs from 27 contributors.
- Dark mode with system / light / dark toggle. (#259)
- Visible conversation timestamps. (#120)
- React artifact output support. (#121)
- Preview comment attachments. (#284)
- Auto-detect OpenClaude as a fallback for Claude Code. (#263)
- Standardize agent communication via stdin and remove Windows-specific shims. (#258)
- xAI Grok Imagine integration covering image, video, and native audio. (#276)
kamieditorial paper design system with deck starter. (#226)html-pptskill (lewislulu/html-ppt-skill) with 15 per-template Examples cards. (#193)design-briefskill with structured I-Lang input format. (#184)- Brand-agnostic craft references and Refero-derived lint rules. (#225)
- 11 HyperFrames video prompt templates and media generation README section. (#227)
- Three Kingdoms ARPG Seedance 2.0 video templates (3). (#212)
- Three Kingdoms ARPG gameplay screenshot templates (3). (#207)
- Otaku-dance choreography breakdown infographic template. (#209)
- Anime fighting game screenshot template. (#208)
--prodflag andOD_HOSTfor headless server deployment intools-dev. (#222)- GitHub CI workflow. (#271)
- Daemon
kindFor/mimeForfile classifier tests. (#269)
- Hungarian (
hu) UI locale. (#288) - Polish (
pl) UI locale. (#273) - Korean (
ko) UI locale. (#253) - Turkish (
tr) UI locale. (#233)
- Image / video projects now pick from prompt templates (not design systems). (#192)
- Optimize Electron release artifact size. (#249)
- Restore
startServerPromise contract — returnurl/{ url, server }. (#268) - Emit
tool_usefromtool_execution_startin pi-rpc. (#186) - Clamp Codex reasoning effort to model-supported values. (#223)
- Deliver Claude Code prompt via stdin to avoid spawn
E2BIG/ENAMETOOLONG. (#143) - Include
package.jsonin tarball so packaged app reports correct version. (#260) - Treat
.pyfiles as previewable code in Design Files. (#261) OD_DAEMON_URLuses port 0 instead of actual allocated port (now reports the real port). (#240)- Quote agent bin path when spawning with
shell:trueon Windows. (#232) - Make
max_tokensconfigurable. (#78)
- Suppress hydration warning on
<body>. (#248) - Fix language dropdown overflow in Settings modal. (#281, #287)
- Add scroll to Settings language menu when it overflows view. (#247)
- Preserve deck preview pagination per file. (#119)
- Fix deck preview pagination controls. (#112)
- Use junction instead of dir symlink on Windows in
tools-dev. (#231)
- Replace hardcoded
Claudewith助手in zh-TW assistant role copy. (#262)
- Traditional Chinese (繁體中文) README. (#194)
- Auto-generated metrics SVG updates. (#228, #241)
- Fix metrics workflow protected branch updates. (#219)
0.1.0 - 2026-05-01
First public release of Open Design — a local-first, open-source alternative to Anthropic's Claude Design. It detects your installed code-agent CLI, runs design skills against curated design systems, and streams artifacts into a sandboxed in-app preview.
- Multi-agent runtime detection and dispatch: Claude Code, Codex, Cursor, Gemini CLI, OpenCode, Qwen, GitHub Copilot CLI, Hermes, Kimi CLI, Pi, and Kiro. (#28, #71, #117, #185)
- Per-CLI model picker for local agents. (#14)
- OpenAI-compatible provider support and Anthropic-compatible stream proxy for non-native providers. (#80, #180)
- App version awareness shared across daemon and web. (#204)
- 72 brand-grade design systems and 31 composable skills, including Xiaohongshu and Replit Deck (8 themes). (#24, #74)
- 57 DESIGN.md specs imported from awesome-design-skills. (#92)
- Dance storyboard and ancient-China MMO HUD prompt templates. (#187)
- Artifact platform foundation with sandboxed in-app preview. (#68)
- First-class SVG and Markdown artifact renderers / viewer. (#73, #177)
- HTML preview support for relative-asset references. (#156)
- Document preview support for uploaded files and multi-file design uploads. (#31, #63)
- Claude Design
.zipimport. (#46) - Image / video / audio media surfaces with unified
od media generatedispatcher. (#12)
- Mac arm64 packaged runtime with signed/notarized DMG + update ZIP and beta release flow. (#170)
- Windows x64 NSIS installer (unsigned beta) and release assets. (#191)
- Vercel self-deploy flow with
vercel.jsonconfiguration. (#167, #169)
- UI locales: zh-CN, zh-TW, en, ja, de, es-ES, ru, fa, pt-BR. (#79, #80, #155, #159, #182, #190, #197)
- Improved language switcher UI. (#107)
tools-dev/tools-packworkspace tooling for development and packaging, with native addon diagnostics and improved web startup flow. (#127, #128, #153)dev:allauto-switches to a free port when defaults are busy. (#9)- UI end-to-end automation suite and reporting under
apps/e2e. (#64, #102) - Frontend toolchain migrated from Vite to Next.js 16 App Router. (#66)
- Project code migrated to TypeScript with shared contracts. (#118)
- Refreshed desktop integration control plane. (#123)
- Star-us prompt to surface GitHub repo. (#5)
- Chat runs survive web reconnects. (#146)
- Daemon project-root resolution when launched from src via tsx. (#162)
- SSE keepalive behind nginx. (#111)
- Standalone pnpm binary supported in postinstall; install toolchain pinned. (#35, #151)
- Surface unfinished todo runs in chat. (#76)
- Spawn agents via resolved absolute path on Windows. (#13)
- Deliver prompts via stdin for non-Claude agents to avoid
spawn ENAMETOOLONG. (#15) - Mitigate Windows
ENAMETOOLONGand fix daemon crash on cleanup. (#75) - Fix
PROMPT_TEMP_FILE()call and Claude Code stdin delivery on Windows. (#97) - Normalize web dev tsconfig paths on Windows for
tools-dev. (#174) - Support Claude Code CLI <1.0.86 (avoid
--include-partial-messages, parse assistant wrapper text). (#34)
- CORS header on raw project file endpoint. (#140)
- Preserve non-ASCII filenames on multipart upload. (#166)
- Stop passing literal dash to
cursor-agent. (#160) - Non-interactive permissions for agent CLIs in web UI. (#26)
- Codex plugin disable env. (#133)
- Codex assistant agent labels. (#70)
- Welcome dialog: stop overwriting user's agent pick on Save. (#4)
- Allow Claude Code to read skill seeds and design-system specs. (#7)
- Question form checkbox selection limits enforced. (#81)
- SettingsDialog content overflow + scrolling, refactored layout and modal styling. (#83, #88)
- Duplicate
H.heading indiscovery.ts(→I.). (#87) - guizang-ppt: sync host slide counter on transform-paginated decks. (#19)
- Toolbar button text wrapping prevented for CJK languages. (#178)
- PreviewModal exits fullscreen on first Esc. (#168)
- Dev indicator moved to bottom-right corner. (#108)
- Design Files: align upload picker with dropzone, neutral agent copy, remove unsupported Figma copy. (#199, #200, #201)
- Web locale registry test includes Japanese. (#202)