name: effect-ts description: "Comprehensive Effect-TS development guide for TypeScript, focused on Effect v4 (the recommended default) with full v3 (stable) support for existing codebases. Use when building, debugging, reviewing, or generating Effect code: typed errors, fibers, Context/Layers, Scope, Schedule, streams, Schema, observability, HTTP, Config, SQL, CLI, RPC, STM, and Effect AI. Includes exhaustive wrong-vs-correct API tables to prevent hallucinated Effect code. Triggers when code imports from 'effect', '@effect/platform', '@effect/ai', or '@effect/sql', or the user mentions Effect-TS, functional TypeScript, Context, Layer, or Schema from Effect." metadata: version: "0.5.0" upstream: "effect@4.0.0-beta.78" openclaw: homepage: https://github.com/tenequm/skills/tree/main/skills/effect-ts emoji: "๐" envVars: - name: OPENAI_API_KEY required: false description: OpenAI API key for Effect AI examples using the OpenAI provider. - name: ANTHROPIC_API_KEY required: false description: Anthropic API key for Effect AI examples using the Anthropic provider.
Effect-TS
Effect is a TypeScript library for building production-grade software with typed errors, structured concurrency, dependency injection, and built-in observability.
Version Detection
Before writing Effect code, detect which version the user is on:
# Check installed version
cat package.json | grep '"effect"'
- v4.x (recommended, the direction Effect is heading):
Context.Service,Effect.catch,Effect.forkChild,Schema.TaggedErrorClass - v3.x (stable, still common in production):
Context.Tag,Effect.catchAll,Effect.fork,Data.TaggedError
Note: v4 beta briefly used a
ServiceMapmodule, renamed back toContexton 2026-04-07 (PR #1961). If you seeServiceMap.*in any doc or older beta code, it is the currentContext.*. Both v3 and v4 importContextfrom"effect"; the exports inside differ (Context.Servicein v4 vsContext.Tagin v3).
Prefer v4 for new projects - it's where Effect is going. In an existing codebase, match the installed version: don't rewrite v3 code in v4 syntax unless asked. If the version is genuinely unclear, default to v4 and say so. v4 is still in beta, so pin an exact version (4.0.0-beta.x) and expect occasional API churn.
Primary Documentation Sources
v4 (primary):
- https://github.com/Effect-TS/effect-smol (v4 source + migration guides)
- https://github.com/Effect-TS/effect-smol/blob/main/LLMS.md (v4 LLM guide)
v3 (for existing codebases):
- https://effect.website/docs (v3 stable docs)
- https://effect.website/llms.txt (LLM topic index)
- https://effect.website/llms-full.txt (full docs for large context)
Both versions:
- https://tim-smart.github.io/effect-io-ai/ (concise API list)
AI Guardrails: Critical Corrections
LLM outputs frequently contain incorrect Effect APIs. Verify every API against the reference docs before using it.
Common hallucinations (both versions):
| Wrong (AI often generates) | Correct |
|---|---|
Effect.cachedWithTTL(...) |
Cache.make({ capacity, timeToLive, lookup }) |
Effect.cachedInvalidateWithTTL(...) |
cache.invalidate(key) / cache.invalidateAll() |
Effect.mapError(effect, fn) |
Effect.mapError(fn) in pipe, or use Effect.catchTag |
import { Schema } from "@effect/schema" |
import { Schema } from "effect" (v3.10+ and all v4) |
import { JSONSchema } from "@effect/schema" |
import { JSONSchema } from "effect" (v3.10+) |
| JSON Schema Draft 2020-12 | Effect Schema generates Draft-07 |
| "thread-local storage" | "fiber-local storage" via FiberRef (v3) / Context.Reference (v4) |
| fibers are "cancelled" | fibers are "interrupted" |
| all queues have back-pressure | only bounded queues; sliding/dropping do not |
new MyError("message") |
new MyError({ message: "..." }) (Schema errors take objects) |
v3-specific hallucinations:
| Wrong | Correct (v3) |
|---|---|
Effect.Service (function call) |
class Foo extends Effect.Service<Foo>()("id", {}) |
Effect.match(effect, { ... }) |
Effect.match(effect, { onSuccess, onFailure }) |
Effect.provide(layer1, layer2) |
Effect.provide(Layer.merge(layer1, layer2)) |
v4-specific hallucinations (AI may mix v3/v4):
| Wrong (v3 API used in v4 code) | Correct (v4) |
|---|---|
Context.Tag("X") (v3 shape) |
Context.Service<X>(id) or class syntax |
ServiceMap.Service / ServiceMap.Reference |
Renamed back to Context.Service / Context.Reference on 2026-04-07 |
Effect.catchAll(fn) |
Effect.catch(fn) |
Effect.fork(effect) |
Effect.forkChild(effect) |
Effect.forkDaemon(effect) |
Effect.forkDetach(effect) |
Data.TaggedError |
Schema.TaggedErrorClass |
FiberRef.get(ref) |
yield* References.X (a Context.Reference) |
yield* ref (Ref as Effect) |
yield* Ref.get(ref) (Ref is no longer an Effect) |
yield* fiber (Fiber as Effect) |
yield* Fiber.join(fiber) (Fiber is no longer Effect) |
Logger.Default / Logger.Live |
Logger.layer (v4 naming convention) |
Schema.TaggedError |
Schema.TaggedErrorClass |
Schema.makeUnsafe(input) |
Schema.make(input) (throws SchemaError); also Schema.makeOption, Schema.makeEffect |
ParseResult (from "effect") |
SchemaIssue module + SchemaError class; narrow with Schema.isSchemaError |
HttpApiEndpoint.get(n, p).pipe(HttpApiEndpoint.setPath(...), setPayload(...), setSuccess(...)) |
HttpApiEndpoint.get(n, p, { params, query, payload, success, error }) (object-option form) |
Otlp.layer({ url, serviceName }) |
OtlpTracer.layer({ url, resource: { serviceName } }) + OtlpSerialization.layerJson + FetchHttpClient.layer |
import { HttpApi } from "@effect/platform" (v4) |
import { HttpApi } from "effect/unstable/httpapi" |
| HttpApi endpoint schema errors are typed errors by default | Since PR #2057 (2026-04-20) they default to defects unless transformed |
Read references/llm-corrections.md for the exhaustive corrections table.
Progressive Disclosure
Read only the reference files relevant to your task:
- Error modeling or typed failures โ
references/error-modeling.md - Services, DI, or Layer wiring โ
references/dependency-injection.md - Retries, timeouts, or backoff โ
references/retry-scheduling.md - Fibers, forking, or parallel work โ
references/concurrency.md - Request batching, N+1 elimination, DataLoader pattern โ
references/concurrency.md - Multi-provider fallback (
ExecutionPlan) โreferences/effect-ai.md/references/retry-scheduling.md - Streams, queues, or SSE โ
references/streams.md - Resource lifecycle or cleanup โ
references/resource-management.md - Refreshable values (rotating credentials, polled config) โ
references/resource-management.md - Schema validation or decoding โ
references/schema.md - Branded / nominal types (
Brand) โreferences/schema.md - Logging, metrics, or tracing โ
references/observability.md - HTTP clients or API calls โ
references/http.md - HTTP API servers โ
references/http.md(covers both client and server) - File uploads / multipart form-data โ
references/http.md - LLM/AI integration โ
references/effect-ai.md - Configuration, env vars, secrets โ
references/configuration.md - SQL / database access โ
references/sql.md - Command-line apps โ
references/cli.md - Typed client/server RPC โ
references/rpc.md - Sharded entities, durable workflows, event sourcing โ
references/distributed.md - Transactional state (STM,
Tx*) โreferences/stm.md - Date/time handling โ
references/datetime.md - Immutable nested updates (optics) โ
references/optics.md - Pattern matching (
Match) โreferences/core-patterns.md - Pooling resources (
Pool) โreferences/resource-management.md - Fiber sets, SubscriptionRef, worker threads โ
references/concurrency.md - Testing Effect code โ
references/testing.md - Property-based testing / generating data from schemas โ
references/testing.md - Migrating from async/await โ
references/migration-async.md - Migrating from v3 to v4 โ
references/migration-v4.md - Core types, gen, pipe, running โ
references/core-patterns.md - Full wrong-vs-correct API table โ
references/llm-corrections.md
Core Workflow
- Detect version from
package.jsonbefore writing any code - Clarify boundaries: identify where IO happens, keep core logic as
Effectvalues - Choose style: use
Effect.genfor sequential logic, pipelines for simple transforms. In v4, preferEffect.fn("name")for named functions - Model errors explicitly: type expected errors in the
Echannel; treat bugs as defects - Model dependencies with services and layers; keep interfaces free of construction logic
- Manage resources with
Scopewhen opening/closing things (files, connections, etc.) - Provide layers and run effects only at program edges (
NodeRuntime.runMainorManagedRuntime) - Verify APIs exist before using them - consult https://tim-smart.github.io/effect-io-ai/ or source docs
Starter Function Set
Start with these ~20 functions (the official recommended set):
Creating effects: Effect.succeed, Effect.fail, Effect.sync, Effect.tryPromise
Composition: Effect.gen (+ Effect.fn in v4), Effect.andThen, Effect.map, Effect.tap, Effect.all
Running: Effect.runPromise, NodeRuntime.runMain (preferred for entry points)
Error handling: Effect.catchTag, Effect.catch (v4) / Effect.catchAll (v3), Effect.orDie
Resources: Effect.acquireRelease, Effect.acquireUseRelease, Effect.scoped
Dependencies: Effect.provide, Effect.provideService
Key modules: Effect, Schema, Layer, Option, Result (v4) / Either (v3), Array, Match
DI (v4): Context.Service, Context.Reference, Layer.effect, Effect.fn("name")
DI (v3): Context.Tag, Context.Reference
Import Patterns
Always use barrel imports from "effect":
import { Context, Effect, Schema, Layer, Option, Stream } from "effect"
For companion packages, import from the package name. v3 and v4 differ here:
// v4 (recommended) - platform transports still separate, but HttpApi / observability
// moved under effect/unstable/*
import { NodeRuntime } from "@effect/platform-node"
import { FetchHttpClient } from "effect/unstable/http"
import { HttpApi, HttpApiEndpoint, HttpApiGroup, HttpApiBuilder, HttpApiScalar } from "effect/unstable/httpapi"
import { OtlpLogger, OtlpSerialization, OtlpTracer } from "effect/unstable/observability"
// v3 (stable) companion packages
import { NodeRuntime } from "@effect/platform-node"
import { HttpClient } from "@effect/platform"
import { NodeSdk } from "@effect/opentelemetry"
Avoid deep module imports (effect/Effect) unless your bundler requires it for tree-shaking.
Output Standards
- Show imports in every code example
- Prefer
Effect.gen(imperative) for multi-step logic; pipelines for transforms - In v4, use
Effect.fn("name")instead of bareEffect.genfor named functions; useEffect.fnUntracedfor internal helpers that don't need a span/stack-frame - Never call
Effect.runPromise/Effect.runSyncinside library code - only at program edges - Use
NodeRuntime.runMainfor CLI/server entry points (handles SIGINT gracefully) - Use
ManagedRuntimewhen integrating Effect into non-Effect frameworks (Hono, Express, etc.) - Always
return yield*when raising an error in a generator (ensures TS understands control flow) - Avoid point-free/tacit usage: write
Effect.map((x) => fn(x))notEffect.map(fn)(generics get erased) - Keep dependency graphs explicit (services, layers, tags)
- State the
Effect<A, E, R>shape when it helps design decisions
Agent Quality Checklist
Before outputting Effect code, verify:
- Every API exists (check against tim-smart API list or source docs)
- Imports are from
"effect"(not@effect/schema,@effect/io, etc.) - Version matches the user's codebase (v3 vs v4 syntax)
- Expected errors are typed in
E; unexpected failures are defects -
run*is called only at program edges, not inside library code - Resources opened with
acquireReleaseare wrapped inEffect.scoped - Layers are provided before running (no missing
Rrequirements) - Generator bodies use
yield*(notyieldwithout*) - Error raises in generators use
return yield*pattern