@chax-at/transactional-prisma-testing
This package provides an easy way to run test cases inside a single database transaction that will
be rolled back after each test. This allows fast test execution while still providing the same source database state for each test.
It also allows parallel test execution against the same database.
Prerequisites
- You are using Prisma 4.7.0 or later in your project.
- You are using PostgreSQL (other DBs might work but have not been tested).
- You are not using Fluent API.
Usage
Install the package by running
npm i -D @chax-at/transactional-prisma-testing
Note that this will install the package as a dev dependency, intended to be used during tests only (remove the -D
if you want to use it outside of tests).
Example
The following example for a NestJS project shows how this package can be used.
It is however possible to use this package with every testing framework, just check out the documentation below and adapt the code accordingly.
You can simply replace the PrismaService
with PrismaClient
if you are not using NestJS.
import { PrismaTestingHelper } from '@chax-at/transactional-prisma-testing';
let prismaTestingHelper: PrismaTestingHelper<PrismaService> | undefined;
let prismaService: PrismaService;
async function before(): Promise<void> {
if(prismaTestingHelper == null) {
const originalPrismaService = new PrismaService();
prismaTestingHelper = new PrismaTestingHelper(originalPrismaService);
prismaService = prismaTestingHelper.getProxyClient();
}
await prismaTestingHelper.startNewTransaction();
}
function after(): void {
prismaTestingHelper?.rollbackCurrentTransaction();
}
function getMockRootModuleBuilder(): TestingModuleBuilder {
return Test.createTestingModule({
imports: [AppModule],
}).overrideProvider(PrismaService)
.useValue(prismaService);
}
PrismaTestingHelper
The PrismaTestingHelper
provides a proxy to the prisma client and manages this proxy.
constructor(prismaClient: T)
Create a single PrismaTestingHelper
per test runner (or a single global one if tests are executed sequentially).
The constructor parameter is the original PrismaClient
that will be used to start transaction.
Note that it is possible to use any objects that extend the PrismaClient, e.g. a NestJS PrismaService.
All methods that don't exist on the prisma transaction client will be routed to this original object (except for $transaction
calls).
getProxyClient(): T
This method returns a Proxy
to the PrismaClient that will execute all calls inside a transaction.
You can save and cache this reference, all calls will always be executed inside the newest transaction.
This allows you to e.g. start your application once with the ProxyClient instead of the normal client
and then execute all test cases, and all calls will always be routed to the newest transaction.
It is usually enough to fetch this once and then use this reference everywhere.
startNewTransaction(opts?: { timeout?: number; maxWait?: number}): Promise<void>
Starts a new transaction. Must be called before each test (and should be called before any query on the proxy client is executed).
You can provide timeout values - note that the transaction timeout
must be long enough so that your
whole test case will be executed during this timeout.
You must call rollbackCurrentTransaction
before calling this method again.
rollbackCurrentTransaction(): void
Ends the currently active transaction. Must be called after each test so that a new transaction can be started.
Limitations / Caveats
- Fluent API is not supported.
- Sequences (auto increment IDs) are not reset when transaction are rolled back. If you need specific IDs in your tests, you can
reset all sequences by using SETVAL before each test.
@default(now())
in your schema (e.g. for a createdAt
date) or similar functionality (e.g. CURRENT_TIMESTAMP()
in PostgreSQL) will always use the start of transaction timestamp. Therefore, all createdAt
-timestamps will have the same value during a test (because they are executed in the same transaction). If this behavior is problematic (e.g. because you want to find the latest entry by creation date), then you can use @default(dbgenerated("statement_timestamp()"))
instead of @default(now())
(if you do not rely on the default "createdAt
= time at start of transaction instead of statement" behavior)
- Transactions in test cases are completely supported by using PostgreSQL Savepoints.
- If you are using parallel transactions (e.g.
await Promise.all(/* Multiple calls that will each start transactions */);
), then they will
automatically be executed sequentially (otherwise, Savepoints wouldn't work). You do not have to change your code for this to work.