home / skills / andrueandersoncs / claude-skill-effect-ts / error-management

error-management skill

/skills/error-management

This skill helps you reason about Effect error handling, distinguish failures, and implement typed, recoverable errors and defects.

npx playbooks add skill andrueandersoncs/claude-skill-effect-ts --skill error-management

Review the files below or copy the command above to add this skill to your agents.

Files (1)
SKILL.md
7.0 KB
---
name: Error Management
description: This skill should be used when the user asks about "Effect errors", "typed errors", "error handling", "Effect.catchAll", "Effect.catchTag", "Effect.mapError", "Effect.orElse", "error accumulation", "defects vs errors", "expected errors", "unexpected errors", "sandboxing", "retrying", "timeout", "Effect.cause", "TaggedError", "Schema.TaggedError", or needs to understand how Effect handles failures in the error channel.
version: 1.0.0
---

# Error Management in Effect

## Overview

Effect distinguishes between two types of failures:

1. **Expected Errors (Recoverable)** - Represented in the `Error` type parameter, tracked at compile time
2. **Defects (Unexpected/Unrecoverable)** - Runtime exceptions, bugs, not in type signature

```typescript
Effect<Success, Error, Requirements>
//              ^^^^^ Expected errors live here
```

## Creating Typed Errors

### Using Schema.TaggedError (Recommended)

```typescript
import { Schema, Effect } from "effect"

class UserNotFound extends Schema.TaggedError<UserNotFound>()(
  "UserNotFound",
  { userId: Schema.String }
) {}

// Note: Schema.Unknown is semantically correct here because `cause` captures
// arbitrary caught exceptions whose type is genuinely unknown at the domain level.
// This is NOT type weakening - JavaScript exceptions can be any value.
class NetworkError extends Schema.TaggedError<NetworkError>()(
  "NetworkError",
  { cause: Schema.Unknown }
) {}

const getUser = (id: string): Effect.Effect<User, UserNotFound | NetworkError> =>
  Effect.gen(function* () {
    // ...implementation
    return yield* Effect.fail(new UserNotFound({ userId: id }))
  })
```

### Using Effect.fail

```typescript
const divide = (a: number, b: number) =>
  b === 0
    ? Effect.fail(new DivisionByZero())
    : Effect.succeed(a / b)
```

## Catching and Recovering from Errors

### catchAll - Catch All Errors

```typescript
program.pipe(
  Effect.catchAll((error) =>
    Effect.succeed("fallback value")
  )
)
```

### catchTag - Catch Specific Error by Tag

```typescript
const program = getUser(id).pipe(
  Effect.catchTag("UserNotFound", (error) =>
    Effect.succeed(defaultUser)
  ),
  Effect.catchTag("NetworkError", (error) =>
    Effect.retry(Schedule.exponential("1 second"))
  )
)
```

### catchTags - Handle Multiple Error Types

```typescript
const program = getUser(id).pipe(
  Effect.catchTags({
    UserNotFound: (error) => Effect.succeed(defaultUser),
    NetworkError: (error) => Effect.fail(new ServiceUnavailable())
  })
)
```

### orElse - Provide Fallback Effect

```typescript
const primary = fetchFromPrimary()
const fallback = fetchFromBackup()

const resilient = primary.pipe(
  Effect.orElse(() => fallback)
)
```

### orElseSucceed - Provide Fallback Value

```typescript
const program = fetchConfig().pipe(
  Effect.orElseSucceed(() => defaultConfig)
)
```

## Transforming Errors

### mapError - Transform Error Type

```typescript
const program = rawApiCall().pipe(
  Effect.mapError((error) => new ApiError({ cause: error }))
)
```

### mapBoth - Transform Both Success and Error

```typescript
const program = effect.pipe(
  Effect.mapBoth({
    onError: (e) => new WrappedError({ cause: e }),
    onSuccess: (a) => a.toUpperCase()
  })
)
```

## Error Accumulation

When running multiple effects, collect all errors instead of failing fast:

### Using Effect.all with mode: "either"

```typescript
const results = yield* Effect.all(
  [effect1, effect2, effect3],
  { mode: "either" }
)
```

### Using Effect.partition

```typescript
const [failures, successes] = yield* Effect.partition(
  items,
  (item) => processItem(item)
)
```

### Using Effect.validate

```typescript
const result = yield* Effect.validate(
  [check1, check2, check3],
  { concurrency: "unbounded" }
)
```

## Defects (Unexpected Errors)

Defects are bugs/unexpected failures not tracked in types:

```typescript
const defect = Effect.die(new Error("Unexpected!"))

const program = effect.pipe(Effect.orDie)

const sandboxed = Effect.sandbox(program)
```

### Cause - Full Error Information

The `Cause` type contains complete failure information:

```typescript
import { Cause, Match } from "effect"

// In sandbox, you get full Cause - use Match for handling
const handled = Effect.sandbox(program).pipe(
  Effect.catchAll((cause) =>
    Match.value(cause).pipe(
      Match.when(Cause.isFailure, () => {
        // Expected error
        return Effect.succeed(fallback)
      }),
      Match.when(Cause.isDie, () => {
        // Defect - log and recover
        return Effect.succeed(fallback)
      }),
      Match.when(Cause.isInterrupt, () => {
        // Interruption
        return Effect.succeed(fallback)
      }),
      Match.orElse(() => Effect.succeed(fallback))
    )
  )
)
```

## Retrying

```typescript
import { Schedule } from "effect"

const resilient = effect.pipe(
  Effect.retry(
    Schedule.exponential("100 millis").pipe(
      Schedule.jittered,
      Schedule.compose(Schedule.recurs(5))
    )
  )
)

// Retry with condition - use Match.tag for error type checking
const conditional = effect.pipe(
  Effect.retry({
    schedule: Schedule.recurs(3),
    while: (error) =>
      Match.value(error).pipe(
        Match.tag("NetworkError", () => true),
        Match.orElse(() => false)
      )
  })
)
```

## Timeouts

```typescript
const withTimeout = effect.pipe(
  Effect.timeout("5 seconds")
)

const failOnTimeout = effect.pipe(
  Effect.timeoutFail({
    duration: "5 seconds",
    onTimeout: () => new TimeoutError()
  })
)
```

## Error Matching Patterns

### Using Effect.match

```typescript
const result = yield* effect.pipe(
  Effect.match({
    onFailure: (error) => `Failed: ${error.message}`,
    onSuccess: (value) => `Success: ${value}`
  })
)
```

### Using Effect.matchEffect

```typescript
const result = yield* effect.pipe(
  Effect.matchEffect({
    onFailure: (error) => logError(error).pipe(Effect.as("failed")),
    onSuccess: (value) => logSuccess(value).pipe(Effect.as("success"))
  })
)
```

## Best Practices

1. **Use TaggedError for all domain errors** - Enables `catchTag` pattern matching
2. **Keep error channel for recoverable errors** - Use defects for bugs
3. **Transform errors at boundaries** - Map low-level errors to domain errors
4. **Use typed errors generously** - The compiler tracks them for free
5. **Accumulate validation errors** - Don't fail fast when validating
6. **Only use Schema.Unknown for genuinely untyped values** - The `cause` field on error types is the canonical example (caught JS exceptions can be any value). Never use Schema.Unknown or Schema.Any for fields whose shape you can describe - define proper schemas instead.

## Additional Resources

For comprehensive error management documentation, consult `${CLAUDE_PLUGIN_ROOT}/references/llms-full.txt`.

Search for these sections:
- "Expected Errors" for creating typed errors
- "Error Accumulation" for collecting multiple errors
- "Sandboxing" for handling defects
- "Retrying" for retry policies
- "Timing Out" for timeout patterns
- "Two Types of Errors" for error philosophy

Overview

This skill explains how Effect models and handles failures, distinguishing typed recoverable errors from runtime defects. It shows how to create tagged domain errors, catch and transform errors, accumulate failures, and handle defects via sandboxing, retrying, and timeouts. The guidance focuses on practical APIs like Effect.fail, Effect.catchTag, Effect.mapError, Effect.retry, and Effect.sandbox.

How this skill works

Effect tracks expected, recoverable errors in the effect's Error type parameter and treats unexpected runtime failures as defects not included in the signature. You create typed domain errors with Schema.TaggedError or use Effect.fail for simple cases. Use catchTag/catchAll/orElse to recover, mapError/mapBoth to transform errors, and sandbox to surface full Cause information for handling defects. Retry, timeout, and accumulation primitives let you control retry policies, time-bound effects, and collect multiple errors.

When to use it

  • When you need compile-time, typed domain errors instead of throwing exceptions
  • When you want to match and recover from specific error types (catchTag)
  • When you must retry transient failures or apply timeouts
  • When collecting validation or parallel operation errors instead of failing fast
  • When diagnosing or handling defects using sandboxed Cause information

Best practices

  • Model domain errors with Schema.TaggedError so you can use catchTag and pattern matching
  • Keep the error channel for recoverable failures; use defects for bugs and truly unexpected conditions
  • Map low-level errors to domain errors at boundaries so callers see stable types
  • Accumulate validation errors (Effect.validate or Effect.all with mode: 'either') rather than fail fast
  • Use sandbox and Cause matching to distinguish expected errors, defects, and interrupts

Example use cases

  • Return a typed UserNotFound error from a repository and recover with catchTag to supply a default user
  • Retry network calls with an exponential jittered schedule and conditional retry only for NetworkError
  • Transform raw API errors to ApiError with mapError before exposing them to higher layers
  • Run parallel validations and collect all failures using Effect.validate for comprehensive feedback
  • Sandbox a program to inspect Cause and log or recover from defects separately from expected errors

FAQ

When should I use Schema.TaggedError vs Effect.fail?

Use Schema.TaggedError for domain errors you want typed and pattern-matchable; use Effect.fail for simple or ad-hoc failures when typing is not required.

How do I handle unexpected runtime exceptions (defects)?

Use Effect.sandbox to get a Cause, then match on Cause.isDie or Cause.isFailure to log, report, or recover. Treat defects as bugs to fix rather than regular control flow.

How can I retry only certain errors?

Use Effect.retry with a schedule and a while/conditional predicate, matching error tags (e.g., Match.tag('NetworkError')) to decide whether to retry.