effectts

Idiomatic Effect-TS patterns for TypeScript functional programming. USE THIS SKILL WHEN: Writing new Effect services or layers, debugging Effect type errors (layer composition, error types), choosing between Effect patterns, setting up Effect testing. TRIGGERS ON: 'effect pattern', 'Layer.provide', 'Context.Tag', 'effect service', 'effect error', 'effect test', 'idiomatic effect'. NOT FOR: general TypeScript without Effect.

Safety Notice

This listing is imported from skills.sh public index metadata. Review upstream SKILL.md and repository scripts before running.

Copy this and send it to your AI assistant to learn

Install skill "effectts" with this command: npx skills add artimath/effect-skills/artimath-effect-skills-effectts

Idiomatic Effect-TS

Write Effect code that looks like it came from the Effect core team.

Prerequisites (Optional)

Research First (ALWAYS)

1. Effect Docs MCP (preferred for concepts)

If you have effect-mcp configured:

// Search documentation
mcp__effect-docs__effect_docs_search({ query: "Layer composition" })

// Then read specific doc
mcp__effect-docs__get_effect_doc({ documentId: 123 })

Otherwise, check https://effect.website/docs for API reference.

2. Effect Source (preferred for real patterns)

# Clone Effect source for pattern verification (if not already done)
# git clone https://github.com/Effect-TS/effect <effect-repo>

# Service patterns (Context.Tag for libraries)
rg "class .* extends Context\.Tag" <effect-repo>/packages/workflow/src
rg "class .* extends Context\.Tag" <effect-repo>/packages/cluster/src

# Effect.Service for concrete impls/tests
rg "Effect\.Service" <effect-repo>/packages/cluster/src

# Error patterns
rg "Schema\.TaggedError" <effect-repo>/packages/cluster/src

# Layer composition
rg "Layer.provideMerge" <effect-repo>/packages/platform/src

Use parallel searches when patterns are unclear.

Quick Reference

Service Patterns (Choose Based on Use Case)

PatternUse WhenLayer Access
Context.TagLibrary interfaces, multiple implsServiceName.Live
Effect.ServiceConcrete impls, tests, appsServiceName.Default
// Library interface (Context.Tag) - canonical for libraries
export class MyService extends Context.Tag("app/MyService")<MyService, {
  readonly doThing: (x: string) => Effect.Effect<Result, MyError>
}>() {
  static readonly Live: Layer.Layer<MyService, never, Deps> = Layer.effect(...)
}

// Concrete impl (Effect.Service) - convenient for apps/tests
export class AppCache extends Effect.Service<AppCache>()("app/Cache", {
  effect: Effect.gen(function* () {
    return { get: (k) => ..., set: (k, v) => ... }
  }),
  dependencies: [SomeDep.Default]  // auto-composed
}) {}

Domain Errors

export class MyError extends Schema.TaggedError<MyError>()("MyError", {
  reason: Schema.Literal("NotFound", "Invalid"),
  cause: Schema.optional(Schema.Defect)
}) {
  static is(u: unknown): u is MyError {
    return hasProperty(u, "_tag") && isTagged(u, "MyError")
  }
}

Domain Types

// Value objects
export class Address extends Schema.Class<Address>("Address")({
  host: Schema.String,
  port: Schema.Number
}) {}

// Branded IDs — always with REAL constraints
export const UserId = Schema.NonEmptyString.pipe(
  Schema.pattern(/^usr_[a-z0-9]+$/),
  Schema.brand("UserId")
)
export type UserId = typeof UserId.Type

Layer Composition — COMMON GOTCHA

This causes most Effect type errors. Know the difference:

MethodDeps SatisfiedAvailable to ProgramUse When
Layer.provideYesNoInternal layer building
Layer.provideMergeYesYesTests using multiple services
Layer.mergeAllNoYesCombining independent layers
// WRONG - test can't access FileSystem
const TestLive = MyService.Live.pipe(Layer.provide(PlatformLive))
// yield* FileSystem.FileSystem -> ERROR

// RIGHT - both services available
const TestLive = MyService.Live.pipe(Layer.provideMerge(PlatformLive))
// yield* MyService -> works
// yield* FileSystem.FileSystem -> also works!

Error pattern to recognize:

Effect<A, E, SomeService> is not assignable to Effect<A, E, never>

or diagnostic: Missing 'SomeService' in the expected Effect context

-> Means SomeService still required. Use provideMerge instead of provide.

Test Pattern

it.effect("works", () =>
  Effect.gen(function* () {
    const svc = yield* MyService
    expect(yield* svc.doThing("x")).toBe(expected)
  }).pipe(Effect.provide(TestLive))  // <- MUST provide at boundary
)

References (Read When Needed)

TopicFileWhen to Read
Service & Layer patternsreferences/services.mdCreating services, Context.Tag vs Effect.Service, layer composition
Layer dependenciesreferences/layer-dependencies.mdHow services access deps (yield + closure), why NOT factory functions
Schema decision matrixreferences/schema-decision.mdSchema.Class vs Struct vs TaggedClass, branded types, migration patterns
Error handlingreferences/errors.mdSchema.TaggedError, TypeId, refail patterns
Domain typesreferences/domain-types.mdSchema.Class, branded types, TaggedRequest
Testingreferences/testing.md@effect/vitest setup, TDD workflow, test helpers
Collections & Statereferences/data.mdChunk, Option, Ref patterns
Process managementreferences/processes.mdScope, Command, background processes

Anti-Patterns

Anti-PatternFix
Data.TaggedError for domain errorsUse Schema.TaggedError
Layer.mergeAll with dependent layersUse Layer.provideMerge
try/catch in Effect.genUse Effect.try or mapError
Missing Effect.provide(...) in testsAlways provide at test boundary
Plain interfaces for domain typesUse Schema.Struct (or Schema.Class if needs behavior)
String IDs without brandingUse Schema.brand()
Guessing patternsGrep Effect source first
Effect.Service for library interfacesUse Context.Tag
Factory function with service paramYield deps in Effect.gen, close over (see layer-dependencies.md)
Schema.Class for simple DTOsUse Schema.Struct unless needs Equal/Hash
String literal union in structUse Schema.TaggedClass for variants
Phantom type & { _tag }Use Schema.brand() for runtime validation
Bare Schema.String.pipe(Schema.brand(...))Add real constraints: NonEmptyString, pattern(), etc.

Source Transparency

This detail page is rendered from real SKILL.md content. Trust labels are metadata-based hints, not a safety guarantee.

Related Skills

Related by shared tags or category signals.

General

sdd

No summary provided by upstream source.

Repository SourceNeeds Review
Security

effect-deep-audit

No summary provided by upstream source.

Repository SourceNeeds Review
General

effect-best-practices

No summary provided by upstream source.

Repository SourceNeeds Review
Coding

Seerr server manager

CLI for the Seerr media request management API. Search movies and TV shows, create and manage media requests, manage users, track issues, and administer a se...

Registry SourceRecently Updated