Skip to main content

Using Prisma

Prisma is a popular open-source TypeScript ORM. The main idea is to define your application data models (entities) in the Prisma Schema, and then use Prisma to automatically generate schema migrations as well as the corresponding Prisma Client for you to manage your data.

Getting Started

An easy way to get started with Prisma is to bootstrap your application with the Prisma template. To download it, run:

npx -y @dbos-inc/create@latest -t dbos-prisma -n <app-name>

Then, build it, run schema migrations, and start the Prisma sample app:

npm run build
npx dbos migrate
npx dbos start

To see that it's working, visit this URL in your browser: http://localhost:3000/greeting/dbos. You should get this message: Greeting 1: Hello, dbos! Each time you refresh the page, the counter should go up by one.

Setting Up Prisma Schema

In DBOS, the Prisma Schema is defined in the same way as any other Prisma project. In this tutorial we assume it is located at the default path (prisma/schema.prisma), for example:

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

datasource db {
provider = "postgresql"
url = env("DATABASE_URL")
}

model DbosHello {
@@map("dbos_hello")
greeting_id Int @id @default(autoincrement())
greeting String
}

Note that the schema file requires a $DATABASE_URL environment variable to connect to a database. To make it easy, we include in our template a script to automatically generate a prisma/.env file with the correct database URL constructed from your dbos-config.yaml file.

Schema Management

In production scenarios or when using DBOS Cloud, we strongly recommend you manage your database schema using Prisma Migrate. To update your database schema, you first need to edit your Prisma schema file, and then create a new migration by running:

npx prisma migrate dev --name <migration_name>

This automatically generates a new migration (under prisma/migrations/) containing SQL commands to transition your database from its current schema to the schema defined in your latest Prisma schema.

Using Prisma

When using DBOS, database operations are performed in transaction functions. Transaction functions must be annotated with the @DBOS.transaction decorator, and can then access the client with DBOS.prismaClient. Note that as PrismaClient is generated for your application, DBOS.prismaClient should be cast to PrismaClient for tab-completion and type checking. You can make a helper function to do this.

Within the transaction function, access your Prisma client from the .client field of your transaction context. For example:

import { PrismaClient } from "@prisma/client";

function getClient() {return DBOS.prismaClient as PrismaClient;}

export class Hello {
@DBOS.transaction()
static async helloTransaction(name: string) {
const greeting = `Hello, ${name}!`;
const res = await getClient().dbosHello.create({
data: {
greeting: greeting,
},
});
return `Greeting ${res.greeting_id}: ${greeting}`;
}
}

Configuring Prisma

info

If you are using the Prisma template, this is done for you.

To enable Prisma, you must set the app_db_client field in the DBOS configuration file to prisma. You should also configure Prisma migration commands. Here is an example of a configuration file set up for Prisma:

language: node
database:
hostname: 'localhost'
port: 5432
username: 'postgres'
password: ${PGPASSWORD}
connectionTimeoutMillis: 3000
app_db_client: prisma
migrate:
- npx prisma migrate deploy
info

Prisma doesn't support schema migration rollback for successfully applied migrations like other ORMs. Therefore, we omit the rollback configuration here. See the Prisma documentation on rolling back schema changes.

Many Prisma commands, such as those for schema migrations, require a DATABASE_URL environment variable to be correctly set. To avoid managing your configuration in two places, we recommend using this script to automatically generate a prisma/.env file with the correct DATABASE_URL string constructed from your DBOS config:

generate_env.js
const { parseConfigFile } = require('@dbos-inc/dbos-sdk/dist/src/dbos-runtime/config');
const fs = require('node:fs');
const path = require('node:path');

// Load the configuration file
const [dbosConfig, ] = parseConfigFile();

// Write out the .env file
const databaseURL = `postgresql://${dbosConfig.poolConfig.user}:${dbosConfig.poolConfig.password}@${dbosConfig.poolConfig.host}:${dbosConfig.poolConfig.port}/${dbosConfig.poolConfig.database}`;

try {
fs.writeFileSync(path.join(process.cwd(), 'prisma', '.env'), `DATABASE_URL="${databaseURL}"`);
console.log("Wrote database URL to the prisma/.env file.");
} catch (error) {
console.error("Error writing prisma/.env file:", error.message);
}

You also need to generate a Prisma Client from your Prisma schema before compiling your application. Therefore, we recommend that you set the build script in your package.json file as follows:

package.json
{
"scripts": {
"build": "node generate_env.js && npx prisma generate && tsc"
}
}