11 KiB
Schema migration
Practical reference for migrating data types in packages/opencode from
Zod-first definitions to Effect Schema.
Goal
Use Effect Schema as the source of truth for domain models, IDs, inputs, outputs, and typed errors. Prefer native Effect Schema, Standard Schema, and native JSON Schema generation at HTTP, tool, and AI SDK boundaries.
The long-term driver is specs/effect/http-api.md: Schema-first DTOs should
flow through HttpApi / HttpRouter without a Zod translation layer.
Preferred shapes
Data objects
Use Schema.Class for structured data.
export class Info extends Schema.Class<Info>("Foo.Info")({
id: FooID,
name: Schema.String,
enabled: Schema.Boolean,
}) {}
If a schema needs local static helpers, use the two-step withStatics pattern:
export const Info = Schema.Struct({
id: FooID,
name: Schema.String,
}).pipe(withStatics((s) => ({ decode: Schema.decodeUnknownOption(s) })))
Errors
Use Schema.TaggedErrorClass for domain errors.
export class NotFoundError extends Schema.TaggedErrorClass<NotFoundError>()("FooNotFoundError", {
id: FooID,
}) {}
IDs and branded leaf types
Keep branded/schema-backed IDs as Effect schemas.
Refinements
Reuse named refinements instead of re-spelling numeric or string constraints in every schema. Boundary JSON Schema helpers should normalize native Effect JSON Schema output only where a provider requires it.
const PositiveInt = Schema.Number.check(Schema.isInt()).check(Schema.isGreaterThan(0))
const NonNegativeInt = Schema.Number.check(Schema.isInt()).check(Schema.isGreaterThanOrEqualTo(0))
const HexColor = Schema.String.check(Schema.isPattern(/^#[0-9a-fA-F]{6}$/))
Compatibility rule
During migration, route validators, tool parameters, and AI SDK schemas should consume Effect schemas directly or use a narrow boundary helper. Avoid maintaining a second hand-written Zod schema.
The default should be:
- Effect Schema owns the type
- new domain models should not start Zod-first unless there is a concrete boundary-specific need
When Zod can stay
It is fine to keep a Zod-native schema temporarily when:
- the type is only used at an HTTP or tool boundary and is not reused elsewhere
- the validator is part of an existing public API that explicitly accepts Zod
- the migration would force unrelated churn across a large call graph
When this happens, prefer leaving a short note or TODO rather than silently creating a parallel schema source of truth.
Boundary helpers
Use narrow helpers at concrete boundaries instead of a generic Schema → Zod bridge.
- Tool parameters:
ToolJsonSchema.fromSchema(...)andToolJsonSchema.fromTool(...) - Public config/TUI schemas:
packages/opencode/script/schema.ts - AI SDK object generation:
Schema.toStandardSchemaV1(...)plusSchema.toStandardJSONSchemaV1(...)
Plugin tools are the main remaining intentional Zod boundary because the public
plugin API exposes tool.schema = z and args: z.ZodRawShape.
Local DeepMutable<T> in config/config.ts
Schema.Struct produces readonly types. Some consumer code (notably the
Config service) mutates Info objects directly, so a readonly-stripping
utility is needed when casting the derived zod schema's output type.
Types.DeepMutable from effect-smol would be a drop-in, but it widens
unknown to {} in the fallback branch — a bug that affects any schema
using Schema.Record(String, Schema.Unknown).
Tracked upstream as effect:core/x228my: "Types.DeepMutable widens unknown
to {}." Once that lands, the local DeepMutable copy can be deleted and
Types.DeepMutable used directly.
Ordering
Migrate in this order:
- Shared leaf models and
schema.tsfiles - Exported
Info,Input,Output, and DTO types - Tagged domain errors
- Service-local internal models
- Route and tool boundary validators that can switch to native Effect Schema helpers
This keeps shared types canonical first and makes boundary updates mostly mechanical.
Progress tracker
src/config/ ✅ complete
All of packages/opencode/src/config/ has been migrated. The export const <Info|Spec> values are all Effect Schema at source.
A file is considered "done" when:
- its exported schema values (
Info,Input,Event,Definition, etc.) are authored as Effect Schema - any remaining Zod is an explicit boundary compatibility choice, not a hand-written parallel source of truth
Files that meet this bar but still carry a compatibility boundary are checked off with an inline note describing the boundary and what unblocks its removal.
- skills, formatter, console-state, mcp, lsp, permission (leaves), model-id, command, plugin, provider
- server, layout
- keybinds
- permission#Info
- agent
- config.ts root
src/*/schema.ts leaf modules
These are the highest-priority next targets. Each is a small, self-contained schema module with a clear domain.
src/account/schema.tssrc/control-plane/schema.tssrc/permission/schema.tssrc/project/schema.tssrc/provider/schema.tssrc/pty/schema.tssrc/question/schema.tssrc/session/schema.tssrc/storage/schema.tssrc/sync/schema.tssrc/tool/schema.tssrc/util/schema.ts
Session domain
Major cluster. Message + event types flow through the SSE API and every SDK output, so byte-identical SDK surface is critical.
Suggested order for this cluster, starting from the leaves that session.ts
and the SSE/event surface depend on:
src/session/schema.ts✅ already migratedsrc/provider/schema.tsifmessage-v2.tsstill relies on zod-first IDssrc/lsp/*schema leaves needed byLSP.Rangesrc/snapshot/*leaves used bySnapshot.FileDiffsrc/session/message-v2.tssrc/session/message.tssrc/session/prompt.tssrc/session/revert.tssrc/session/summary.tssrc/session/status.tssrc/session/todo.tssrc/session/session.tssrc/session/compaction.ts
Dependency sketch:
session.ts
|- project/schema.ts
|- control-plane/schema.ts
|- permission/schema.ts
|- snapshot/*
|- message-v2.ts
| |- provider/schema.ts
| |- lsp/*
| |- snapshot/*
| |- sync/index.ts
| `- bus/bus-event.ts
|- sync/index.ts
|- bus/bus-event.ts
`- util/update-schema.ts
Working rule for this cluster:
- migrate reusable leaf schemas and nested payload objects first
- migrate aggregate DTOs like
Session.Infoafter their nested pieces exist as named Schema values - leave zod-only event/update helpers in place temporarily when converting them would force unrelated churn across sync/bus boundaries
message-v2.ts first-pass outline:
- Schema-backed imports already available
SessionID,MessageID,PartIDProviderID,ModelID
- Local leaf objects to extract and migrate first
- output format payloads
- common part bases like
PartBase - timestamp/range helper objects like
time.start/end - file/source helper objects
- token/cost/model helper objects
- Part variants built from those leaves
SnapshotPart,PatchPart,TextPart,ReasoningPartFilePart,AgentPart,CompactionPart,SubtaskPart- retry/step/tool related parts
- Higher-level unions and DTOs
FilePartSource- part unions
- message unions and assistant/user payloads
- Errors and event payloads last
NamedError.create(...)shapes can stay temporarily if converting them toSchema.TaggedErrorClasswould force unrelated churnSyncEvent.define(...)andBusEvent.define(...)payloads can use derived.zodat remaining zod-based HTTP/OpenAPI boundaries
Possible later tightening after the Schema-first migration is stable:
-
promote repeated opaque strings and timestamp numbers into branded/newtype leaf schemas where that adds domain value without changing the wire format
-
src/session/compaction.ts -
src/session/message-v2.ts -
src/session/message.ts -
src/session/prompt.ts -
src/session/revert.ts -
src/session/session.ts -
src/session/status.ts -
src/session/summary.ts -
src/session/todo.ts
Provider domain
src/provider/auth.tssrc/provider/models.tssrc/provider/provider.ts
Tool schemas
Each tool declares its parameters via a zod schema. Tools are consumed by both the in-process runtime and the AI SDK's tool-calling layer, so the emitted JSON Schema must stay byte-identical.
src/tool/apply_patch.tssrc/tool/bash.tssrc/tool/edit.tssrc/tool/glob.tssrc/tool/grep.tssrc/tool/invalid.tssrc/tool/lsp.tssrc/tool/plan.tssrc/tool/question.tssrc/tool/read.tssrc/tool/registry.tssrc/tool/skill.tssrc/tool/task.tssrc/tool/todo.tssrc/tool/tool.tssrc/tool/webfetch.tssrc/tool/websearch.tssrc/tool/write.ts
HTTP route boundaries
The server route tree now lives under src/server/routes/instance/httpapi and
uses Effect HttpApi contracts for request and response schemas. Remaining schema
work is no longer a Hono route migration; it is compatibility cleanup around
derived .zod statics, OpenAPI translation shims, and route groups that still
need explicit SDK-visible error contracts.
Good follow-up targets:
- shrink
public.tslegacy OpenAPI translation shims one SDK-compatible slice at a time - replace production
.zod.safeParse(...)call sites with Effect Schema decoders - remove derived
.zodstatics after their production consumers are gone - declare route-group errors directly instead of relying on compatibility middleware
Everything else
Small / shared / control-plane / CLI. Mostly independent; can be done piecewise.
src/acp/agent.tssrc/agent/agent.tssrc/bus/bus-event.tssrc/bus/index.tssrc/cli/cmd/tui/config/tui-migrate.tssrc/cli/cmd/tui/config/tui-schema.tssrc/cli/cmd/tui/config/tui.tssrc/cli/cmd/tui/event.tssrc/cli/ui.tssrc/command/index.tssrc/control-plane/adapters/worktree.tssrc/control-plane/types.tssrc/control-plane/workspace.tssrc/file/index.tssrc/file/ripgrep.tssrc/file/watcher.tssrc/format/index.tssrc/id/id.tssrc/ide/index.tssrc/installation/index.tssrc/lsp/client.tssrc/lsp/lsp.tssrc/mcp/auth.tssrc/patch/index.tssrc/plugin/github-copilot/models.tssrc/project/project.tssrc/project/vcs.tssrc/pty/index.tssrc/skill/index.tssrc/snapshot/index.tssrc/storage/db.tssrc/storage/storage.tssrc/sync/index.ts— public API (SyncEvent.define) is Schema-first;payloads()still derives zod for the remaining HTTP/OpenAPI boundarysrc/util/fn.tssrc/util/log.tssrc/util/update-schema.tssrc/worktree/index.ts
Notes
- Prefer one canonical schema definition. Avoid maintaining parallel Zod and Effect definitions for the same domain type.
- Keep the migration incremental. Converting the domain model first is more valuable than converting every boundary in the same change.
- Every migrated file should leave the generated SDK output (
packages/sdk/ openapi.jsonandpackages/sdk/js/src/v2/gen/types.gen.ts) byte-identical unless the change is deliberately user-visible.