Using Jest
Jest is a JavaScript testing framework that is widely used in the JavaScript community. It is recommended for beginners because it has the largest community and is the most widely used. Jest is also the officially supported test runner for PicJS.
To get started with Jest, install the relevant packages using your preferred package manager:
npm i -D jest @types/jest @types/node ts-jest
pnpm i -D jest @types/jest @types/node ts-jest
bun add -d jest @types/jest @types/node ts-jest
yarn add -D jest @types/jest @types/node ts-jest
Create a tsconfig.json
file:
{ "compilerOptions": { // enable latest features "lib": ["ESNext"], "target": "ESNext", "module": "ESNext", "moduleDetection": "force", "allowJs": true, // allow importing `.js` from `.ts` "types": ["jest", "node"],
// Bundler mode "moduleResolution": "bundler", "allowImportingTsExtensions": true, "verbatimModuleSyntax": true, "noEmit": true,
// Best practices "strict": true, "skipLibCheck": true, "noUnusedLocals": true, "noUnusedParameters": true, "noFallthroughCasesInSwitch": true,
// Some stricter flags "useUnknownInCatchVariables": true, "noPropertyAccessFromIndexSignature": true }, "include": [ "./src/**/*.ts", "./global-setup.ts", "./global-teardown.ts", "./types.d.ts" ]}
Create a jest.config.ts
file:
import type { Config } from 'jest';
const config: Config = { watch: false, preset: 'ts-jest/presets/js-with-ts', testEnvironment: 'node', globalSetup: '<rootDir>/global-setup.ts', globalTeardown: '<rootDir>/global-teardown.ts', testTimeout: 30_000,};
export default config;
You can also check out the official the ts-jest
documentation for more information on configuring this file.
Then, add a test
script to your package.json
:
{ "scripts": { "test": "jest" }}
The PocketIC server needs to be started before running tests and stopped once they’re finished running. This can be done by creating global-setup.ts
and global-teardown.ts
files in your project’s root directory:
import { PocketIcServer } from '@dfinity/pic';
module.exports = async function (): Promise<void> { const pic = await PocketIcServer.start(); const url = pic.getUrl();
process.env.PIC_URL = url; global.__PIC__ = pic;};
module.exports = async function () { await global.__PIC__.stop();};
To improve type-safety for process.env.PIC_URL
and global.__PIC__
, create a types.d.ts
file:
import { PocketIcServer } from '@dfinity/pic';
declare global { declare var __PIC__: PocketIcServer;
namespace NodeJS { interface ProcessEnv { PIC_URL: string; } }}
Writing tests
Section titled “Writing tests”Jest tests are very similar to tests written with Jasmine, or Vitest so they will feel very familiar to developers who have used these frameworks before.
The basic skeleton of all PicJS tests written with Jest will look something like this:
// Import generated types for your canisterimport { type _SERVICE } from '../../declarations/backend/backend.did';
// Define the path to your canister's WASM fileexport const WASM_PATH = resolve( import.meta.dir, '..', '..', 'target', 'wasm32-unknown-unknown', 'release', 'backend.wasm',);
// The `describe` function is used to group tests together// and is completely optional.describe('Test suite name', () => { // Define variables to hold our PocketIC instance, canister ID, // and an actor to interact with our canister. let pic: PocketIc; let canisterId: Principal; let actor: Actor<_SERVICE>;
// The `beforeEach` hook runs before each test. // // This can be replaced with a `beforeAll` hook to persist canister // state between tests. beforeEach(async () => { // create a new PocketIC instance pic = await PocketIc.create(process.env.PIC_URL);
// Setup the canister and actor const fixture = await pic.setupCanister<_SERVICE>({ idlFactory, wasm: WASM_PATH, });
// Save the actor and canister ID for use in tests actor = fixture.actor; canisterId = fixture.canisterId; });
// The `afterEach` hook runs after each test. // // This should be replaced with an `afterAll` hook if you use // a `beforeAll` hook instead of a `beforeEach` hook. afterEach(async () => { // tear down the PocketIC instance await pic.tearDown(); });
// The `it` function is used to define individual tests it('should do something cool', async () => { const response = await actor.do_something_cool();
expect(response).toEqual('cool'); });});
You can also check out the official Jest getting started documentation for more information on writing tests.