Skip to content

riordanpawley/effect-prisma-generator

 
 

Repository files navigation

Effect Prisma Generator

A Prisma generator that creates a fully-typed, Effect-based service wrapper for your Prisma Client.

📦 Latest Release: v0.6.0 - Migration Guide from v0.5

Features

  • 🚀 Effect Integration: All Prisma operations are wrapped in Effect for robust error handling and composability.
  • 🛡️ Type Safety: Full TypeScript support with generated types matching your Prisma schema.
  • 🧩 Dependency Injection: Integrates seamlessly with Effect's Layer and Context system.
  • 🔍 Error Handling: Automatically catches and wraps Prisma errors into typed PrismaError variants.
  • 📊 Optional Telemetry: Enable operation tracing with enableTelemetry config.

Installation

Install the generator as a development dependency:

npm install -D effect-prisma-generator
# or
pnpm add -D effect-prisma-generator
# or
yarn add -D effect-prisma-generator

Configuration

Add the generator to your schema.prisma file:

// prisma/schema.prisma
generator client {
  provider        = "prisma-client-js"
  output          = "./generated/client"
}

generator effect {
  provider = "effect-prisma-generator"
  output   = "./generated/effect" // relative to the schema.prisma file, e.g. prisma/generated/effect
  clientImportPath = "../client" // relative to the output path ^here (defaults to "@prisma/client")
}

Then run prisma generate to generate the client and the Effect service.

Configuration Options

Option Description Default
output Output directory for generated code (relative to schema.prisma) ../generated/effect
clientImportPath Import path for Prisma Client (relative to output) @prisma/client
errorImportPath Custom error module path (relative to schema.prisma), e.g. ./errors#MyError -
importFileExtension File extension for relative imports (js, ts, or empty) ""
enableTelemetry Wrap operations with Effect.fn() for tracing ("true" or "false") "false"

ESM / Import Extensions

For ESM projects that require explicit file extensions in imports, use importFileExtension:

generator effect {
  provider            = "effect-prisma-generator"
  output              = "./generated/effect"
  clientImportPath    = "../client/index.js"
  errorImportPath     = "./errors#MyPrismaError"  // No extension needed here
  importFileExtension = "js"                       // Generator adds .js to relative imports
}

This will generate imports like:

import { MyPrismaError, mapPrismaError } from "../../errors.js"

Recommended

Add the following to your tsconfig.json:

{
  "compilerOptions": {
    "paths": {
      "@prisma/*": ["./prisma/generated/*"]
    }
  }
}

Then you can import the generated types like this:

import { Prisma } from "@prisma/effect";

Otherwise, you can import the generated types like this (adjust the path accordingly):

import { Prisma } from "../../prisma/generated/effect";

Usage

Quick Start

import { Prisma } from "@prisma/effect";
import { Effect } from "effect";

const program = Effect.gen(function* () {
  const prisma = yield* Prisma;

  const users = yield* prisma.user.findMany({
    where: { active: true },
  });

  return users;
});

// Run with the default layer (Prisma 6)
Effect.runPromise(program.pipe(Effect.provide(Prisma.Live)));

Layer Options

The generator provides several ways to create layers:

import { Prisma, PrismaClient } from "@prisma/effect";
import { Effect, Layer } from "effect";

// 1. Default layer (Prisma 6, no options)
Prisma.Live

// 2. Layer with static options
Prisma.layer({ datasourceUrl: process.env.DATABASE_URL })

// 3. Layer with effectful options (for adapters, config services, etc.)
Prisma.layerEffect(
  Effect.gen(function* () {
    const config = yield* ConfigService;
    return { datasourceUrl: config.databaseUrl };
  })
)

// 4. For Prisma 7 with adapters
Prisma.layerEffect(
  Effect.gen(function* () {
    const pool = yield* PostgresPool;
    const adapter = yield* Effect.sync(() => new PrismaNeon(pool));
    return { adapter };
  })
)

Full Example

import { Prisma } from "./generated/effect";
import { Effect } from "effect";

const program = Effect.gen(function* () {
  const prisma = yield* Prisma;

  // All standard Prisma operations are available
  const users = yield* prisma.user.findMany({
    where: { active: true },
    select: {
      id: true,
      accounts: {
        select: {
          id: true,
        },
      },
    },
  });
  // users: { id: string, accounts: { id: string }[] }[]

  return users;
});

// Run the program
Effect.runPromise(program.pipe(Effect.provide(Prisma.Live)));

API

The generated Prisma service mirrors your Prisma Client API but returns Effect<Success, PrismaError, Requirements> instead of Promises.

Layer Constructors

API Description
Prisma.Live Complete default layer (Prisma 6, no options)
Prisma.layer(opts) Complete layer with PrismaClient options
Prisma.layerEffect(effect) Complete layer with effectful options
Prisma.Default Just the service layer (for advanced composition)
PrismaClient.layer(opts) Just the client layer (for advanced composition)
PrismaClient.layerEffect(effect) Just the client layer with effectful options
PrismaClient.Default Default client layer (for advanced composition)

Error Handling

Operations return typed errors that you can handle with Effect's error handling utilities:

import {
  Prisma,
  PrismaUniqueConstraintError,
  PrismaRecordNotFoundError,
  PrismaForeignKeyConstraintError,
} from "@prisma/effect";

const program = Effect.gen(function* () {
  const prisma = yield* Prisma;

  // Handle specific error types with catchTag
  const user = yield* prisma.user
    .create({ data: { email: "[email protected]" } })
    .pipe(
      Effect.catchTag("PrismaUniqueConstraintError", (error) => {
        console.log(`Duplicate email: ${error.cause.code}`); // P2002
        return Effect.succeed(null);
      }),
    );

  // OrThrow methods return PrismaRecordNotFoundError
  const found = yield* prisma.user
    .findUniqueOrThrow({ where: { id: 999 } })
    .pipe(
      Effect.catchTag("PrismaRecordNotFoundError", () =>
        Effect.succeed(null),
      ),
    );
});

Available error types:

Error Type Prisma Code When it occurs
PrismaUniqueConstraintError P2002 Duplicate unique field
PrismaRecordNotFoundError P2025 findUniqueOrThrow, findFirstOrThrow, update, delete on non-existent
PrismaForeignKeyConstraintError P2003 Invalid foreign key reference
PrismaValueTooLongError P2000 Value exceeds column length
PrismaDbConstraintError P2004 Database constraint violation
PrismaInputValidationError P2005, P2006, P2019 Invalid input value
PrismaMissingRequiredValueError P2011, P2012 Required field is null
PrismaRelationViolationError P2014 Relation constraint violation
PrismaRelatedRecordNotFoundError P2015, P2018 Related record not found
PrismaValueOutOfRangeError P2020 Value out of range
PrismaConnectionError P2024 Connection pool timeout
PrismaTransactionConflictError P2034 Transaction conflict (retry)

Custom Error Mapping

If you want to use your own error type instead of the built-in tagged errors, you can configure errorImportPath in your schema:

generator effect {
  provider         = "effect-prisma-generator"
  output           = "./generated/effect"
  clientImportPath = "../client"
  errorImportPath  = "./errors#MyPrismaError"  // relative to schema.prisma
}

Note: The errorImportPath is relative to your schema.prisma file location, not the output directory. The generator automatically calculates the correct import path for the generated code.

Your error module must export:

  1. The error class - Your custom error type
  2. A mapper function named mapPrismaError - Maps raw errors to your type
// errors.ts
import { Data } from "effect";
import { Prisma } from "@prisma/client";

export class MyPrismaError extends Data.TaggedError("MyPrismaError")<{
  cause: unknown;
  operation: string;
  model: string;
  code?: string;  // You can add custom fields
}> {}

export const mapPrismaError = (
  error: unknown,
  operation: string,
  model: string
): MyPrismaError => {
  // You can inspect the error and add custom handling
  const code = error instanceof Prisma.PrismaClientKnownRequestError
    ? error.code
    : undefined;

  // Option: throw unknown errors as defects
  // if (!(error instanceof Prisma.PrismaClientKnownRequestError)) {
  //   throw error;
  // }

  return new MyPrismaError({ cause: error, operation, model, code });
};

Now all operations will use your MyPrismaError type:

import { Prisma, MyPrismaError } from "./generated/effect";

const program = Effect.gen(function* () {
  const prisma = yield* Prisma;

  // All errors are now MyPrismaError
  yield* prisma.user
    .create({ data: { email: "[email protected]" } })
    .pipe(
      Effect.catchTag("MyPrismaError", (error) => {
        console.log(`Operation: ${error.operation}, Code: ${error.code}`);
        return Effect.succeed(null);
      }),
    );
});

This is useful when:

  • Migrating from an existing codebase that uses a single error type
  • You want to add custom fields/metadata to errors
  • You want control over which errors are recoverable vs defects

Transactions

The generated service includes a $transaction method that allows you to run multiple operations within a database transaction.

const program = Effect.gen(function* () {
  const prisma = yield* Prisma;

  const result = yield* prisma.$transaction(
    Effect.gen(function* () {
      const user = yield* prisma.user.create({ data: { name: "Alice" } });
      const post = yield* prisma.post.create({
        data: { title: "Hello", authorId: user.id },
      });
      return { user, post };
    }),
  );
});

Custom Transaction Options

For transactions that need custom options (isolation level, timeout, etc.), use $transactionWith:

// Custom isolation level
yield* prisma.$transactionWith(
  Effect.gen(function* () {
    const user = yield* prisma.user.create({ data: { name: "Alice" } });
    return user;
  }),
  { isolationLevel: "Serializable", timeout: 10000 }
);

// Point-free style for cleaner composition
import { pipe } from "effect";

const myEffect = Effect.gen(function* () {
  const prisma = yield* Prisma;
  return yield* prisma.user.findMany();
});

// Clean, functional composition!
pipe(
  myEffect,
  prisma.$transaction  // No options? Use $transaction directly!
);

// With options, use $transactionWith
const withSerializable = (eff: Effect.Effect<any, any, any>) =>
  prisma.$transactionWith(eff, { isolationLevel: "Serializable" });

pipe(myEffect, withSerializable);

Available transaction options:

  • isolationLevel: "ReadUncommitted" | "ReadCommitted" | "RepeatableRead" | "Serializable"
  • maxWait: Maximum time (ms) Prisma Client will wait to acquire a transaction
  • timeout: Maximum time (ms) the transaction can run before being canceled

Transaction Rollback Behavior

Any uncaught error in the Effect error channel triggers a rollback:

// Rollback on Effect.fail()
yield* prisma.$transaction(
  Effect.gen(function* () {
    yield* prisma.user.create({ data: { email: "[email protected]" } });
    yield* Effect.fail("Something went wrong"); // Triggers rollback
  }),
);
// User is NOT created

// Rollback on Prisma errors (e.g., findUniqueOrThrow)
yield* prisma.$transaction(
  Effect.gen(function* () {
    yield* prisma.user.create({ data: { email: "[email protected]" } });
    yield* prisma.user.findUniqueOrThrow({ where: { id: 999 } }); // Throws!
  }),
);
// User is NOT created

Catching errors prevents rollback:

yield* prisma.$transaction(
  Effect.gen(function* () {
    yield* prisma.user.create({ data: { email: "[email protected]" } });

    // Catch the error - transaction continues
    yield* prisma.user
      .findUniqueOrThrow({ where: { id: 999 } })
      .pipe(Effect.catchAll(() => Effect.succeed(null)));

    yield* prisma.user.create({ data: { email: "[email protected]" } });
  }),
);
// Both users ARE created

Custom error types are preserved:

class MyError extends Data.TaggedError("MyError")<{ message: string }> {}

const error = yield* prisma
  .$transaction(Effect.fail(new MyError({ message: "oops" })))
  .pipe(Effect.flip);

expect(error).toBeInstanceOf(MyError); // Type is preserved!

Nested Transactions

Nested $transaction calls share the same underlying database transaction. There are no savepoints - all operations run in a single transaction that commits or rolls back together.

yield* prisma.$transaction(
  Effect.gen(function* () {
    yield* prisma.user.create({ data: { name: "Outer" } });

    yield* prisma.$transaction(
      Effect.gen(function* () {
        yield* prisma.user.create({ data: { name: "Inner" } });
      }),
    );

    yield* Effect.fail("Outer failure");
  }),
);
// BOTH users are rolled back

Key Behaviors

Scenario Result
Both succeed All committed
Inner fails (uncaught) All rollback
Inner succeeds, outer fails All rollback
Inner fails (caught), outer succeeds All committed (including inner's data!)

Important: When you catch an inner transaction's error, its writes are NOT rolled back because there are no savepoints. All operations share the same database transaction.

Composable Service Functions

Functions that use $transaction internally work seamlessly when called from an outer transaction:

// Service function with its own transaction
const UserService = {
  createWithProfile: (email: string) =>
    Effect.gen(function* () {
      const prisma = yield* Prisma;
      return yield* prisma.$transaction(
        Effect.gen(function* () {
          const user = yield* prisma.user.create({ data: { email } });
          yield* prisma.profile.create({ data: { userId: user.id } });
          return user;
        }),
      );
    }),
};

// Called standalone - creates its own transaction
yield* UserService.createWithProfile("[email protected]");

// Called inside outer transaction - joins it
yield* prisma.$transaction(
  Effect.gen(function* () {
    yield* UserService.createWithProfile("[email protected]");
    yield* UserService.createWithProfile("[email protected]");
    // If anything fails, both users are rolled back
  }),
);

This pattern allows you to:

  1. Write self-contained service functions that are safe to call standalone
  2. Compose them in outer transactions for end-to-end atomicity
  3. Functions don't need to know if they're inside another transaction

Building Effect Services with Prisma

You can build layered Effect services that wrap Prisma. Transactions work correctly through any level of service composition.

// Level 1: Repository layer
class UserRepo extends Effect.Service<UserRepo>()("UserRepo", {
  effect: Effect.gen(function* () {
    const db = yield* Prisma;
    return {
      create: (email: string, name: string) =>
        db.user.create({ data: { email, name } }),
      findById: (id: number) =>
        db.user.findUnique({ where: { id } }),
    };
  }),
}) {}

class PostRepo extends Effect.Service<PostRepo>()("PostRepo", {
  effect: Effect.gen(function* () {
    const db = yield* Prisma;
    return {
      create: (title: string, authorId: number) =>
        db.post.create({ data: { title, authorId } }),
    };
  }),
}) {}

// Level 2: Domain service composing repositories
class BlogService extends Effect.Service<BlogService>()("BlogService", {
  effect: Effect.gen(function* () {
    const users = yield* UserRepo;
    const posts = yield* PostRepo;
    const db = yield* Prisma;

    return {
      createAuthorWithPost: (email: string, name: string, title: string) =>
        db.$transaction(
          Effect.gen(function* () {
            const user = yield* users.create(email, name);
            const post = yield* posts.create(title, user.id);
            return { user, post };
          }),
        ),
    };
  }),
}) {}

// Wire up the layers
const RepoLayer = Layer.merge(UserRepo.Default, PostRepo.Default).pipe(
  Layer.provide(Prisma.Live),
);
const ServiceLayer = BlogService.Default.pipe(
  Layer.provide(RepoLayer),
  Layer.provide(Prisma.Live),
);

// Use it
const program = Effect.gen(function* () {
  const blog = yield* BlogService;
  return yield* blog.createAuthorWithPost("[email protected]", "Alice", "Hello World");
});

Effect.runPromise(program.pipe(Effect.provide(ServiceLayer)));

Why This Works

You might wonder: if Prisma is captured at layer construction time, how do transactions work?

The key is deferred execution. When you call db.user.create({ data }), it doesn't execute immediately—it returns an Effect that describes what to do:

// Generated code (simplified)
user: {
  create: (args) => Effect.flatMap(PrismaClient, ({ tx: client }) =>
    Effect.tryPromise({ try: () => client.user.create(args), ... })
  )
}

The Effect.flatMap(PrismaClient, ...) defers the lookup of PrismaClient until the Effect actually runs. When $transaction executes an inner effect, it provides a new PrismaClient with the transaction client:

// Inside $transaction (simplified)
effect.pipe(Effect.provideService(PrismaClient, { tx: transactionClient, client }))

So even though you capture db (the Prisma service) at layer construction, the actual database client lookup happens at execution time—inside the transaction scope.

This means:

  • ✅ Services can store references to Prisma at construction
  • ✅ Services can store effect-returning methods (e.g., const createUser = db.user.create)
  • ✅ Transactions work correctly through any number of service layers
  • ✅ Nested $transaction calls properly join the outer transaction

Resource Management

The PrismaClient.layer function uses Layer.scoped with a finalizer to ensure the PrismaClient is properly disconnected when the layer scope ends:

// Generated code (simplified)
export class PrismaClient extends Context.Tag("PrismaClient")<...>() {
  static layer = <T extends ConstructorParameters<typeof BasePrismaClient>[0]>(options: T) =>
    Layer.scoped(
      PrismaClient,
      Effect.gen(function* () {
        const prisma = new BasePrismaClient(options)
        yield* Effect.addFinalizer(() => Effect.promise(() => prisma.$disconnect()))
        return { tx: prisma, client: prisma }
      })
    )
}

This means:

  • The connection is automatically cleaned up when the program completes
  • The connection is cleaned up even if the program fails
  • Each scoped usage gets its own PrismaClient instance
// Connection is automatically managed
const program = Effect.gen(function* () {
  const prisma = yield* Prisma;
  yield* prisma.user.findMany();
  // ... more operations
});

// $disconnect is called automatically when this completes
await Effect.runPromise(
  program.pipe(
    Effect.provide(Prisma.Live),
    Effect.scoped,
  )
);

For long-running applications (like servers), you typically provide the layer once at startup and it stays connected for the lifetime of the application.

Migration from v0.x

If you're upgrading from an earlier version, the old API is still available but deprecated:

Old API (deprecated) New API
PrismaService Prisma
PrismaClientService PrismaClient
makePrismaLayer(opts) PrismaClient.layer(opts)
makePrismaLayerEffect(effect) PrismaClient.layerEffect(effect)
LivePrismaLayer PrismaClient.Default
Layer.merge(LivePrismaLayer, PrismaService.Default) Prisma.Live

The deprecated names will be removed in the next major version.

About

Generate a fully-typed, Effect-native service wrapper for Prisma Client with built-in error handling and transaction support.

Resources

License

Stars

Watchers

Forks

Packages

 
 
 

Contributors

Languages

  • TypeScript 74.6%
  • JavaScript 25.2%
  • Shell 0.2%