Follow the root AGENTS.md first. This file only records module-level boundaries for apps/.
apps/web: Next.js 16 App Router + React 18 web runtime. Entrypoints live inapps/web/app/; the main client shell isapps/web/src/App.tsx. During localtools-devweb runs,apps/web/next.config.tsrewrites/api/*,/artifacts/*, and/frames/*toOD_PORT.apps/daemon: Express + SQLite local daemon andodbin. It owns REST/SSE APIs, agent CLI spawning, skills, design systems, artifact persistence, static serving, and local data under.od/.apps/desktop: Electron shell. Desktop does not guess the web port; it reads runtime status through sidecar IPC and opens the reported web URL.apps/packaged: Thin packaged Electron runtime entry. It starts packaged daemon/web sidecars, registers theod://entry protocol, and delegates desktop host behavior toapps/desktop.
apps/daemon/src/contains only daemon app source.apps/daemon/tests/contains daemon tests.apps/daemon/sidecar/contains the daemon sidecar entry.- CLI/agent argument definition changes belong in
apps/daemon/src/runtimes/defs/; stdout parser changes belong with the matching runtime helpers and parser tests.
- Existing daemon domain endpoints belong in the matching daemon route file; avoid adding route handlers directly to
apps/daemon/src/server.tsunless the route is bootstrap-wide or has no clear domain owner. - New route registrars should be wired into the matching semantic section in
server.ts; keep sections broad and reuse existing sections before adding a new one. - Bootstrap-wide routes describe daemon availability or startup metadata shared by every domain.
/api/healthand/api/versionstay inserver.tsbecause they only report process-level status. - Domain routes describe a product capability or data model.
/api/activebelongs inactive-context-routes.tsbecause transient UI focus is its own domain, while chat routes own persistent conversation and run state. - Add endpoints to an existing route file when they share the same domain language and dependency set. Split a new
*-routes.tsmodule when the endpoint introduces a distinct domain or has little dependency overlap with existing route modules.
- App tests live in each app's
tests/directory, sibling tosrc/; preserve source-relative subpaths insidetests/when useful. - Keep app
src/directories source-only; do not add new*.test.tsor*.test.tsxfiles undersrc/. apps/web/tests/contains web-owned Vitest tests and uses*.test.ts/*.test.tsx.- Playwright UI automation belongs in
e2e/ui/; do not add Playwright suites or UI automation helper scripts underapps/web.
- App business layers must not import sidecar packages or branch on
runtime.mode,namespace,ipc, orsource. - Keep sidecar awareness in
apps/<app>/sidecaror the desktop sidecar entry wrapper.
apps/nextjshas been removed; do not restore it.- Packaged web uses Next.js SSR through the web sidecar; do not put Next output under daemon
OD_RESOURCE_ROOT. - Packaged
OD_RESOURCE_ROOTis only for daemon non-Next read-only resources:skills/,design-systems/, andframes/. - Packaged data/log/runtime/cache paths must be namespace-scoped and must not depend on daemon or web ports.
- Daemon↔web packaged traffic still uses an HTTP origin/port because Next.js dev server and SSR proxy paths assume HTTP origins; switching to Unix sockets would require patching Next internals. The invariant is that data/log/runtime/cache paths never embed ports.
pnpm --filter @open-design/web typecheck
pnpm --filter @open-design/web test
pnpm --filter @open-design/daemon typecheck
pnpm --filter @open-design/daemon test
pnpm --filter @open-design/daemon build
pnpm --filter @open-design/desktop typecheck
pnpm --filter @open-design/desktop build
pnpm --filter @open-design/packaged typecheck
pnpm --filter @open-design/packaged build