| name | jazz-testing |
| description | Use this skill when you need to write, review, or debug automated tests for applications built on the Jazz framework. This skill provides the correct architectural patterns for simulating local-first synchronization and multi-user environments without resorting to invalid mocking strategies. |
Jazz Testing
When to Use This Skill
- Writing unit or integration tests for apps using Jazz
- Simulating synchronisation for testing purposes
- Verifying permissions and security logic
- Testing UI components in various application states (Guest/Anonymous/Authenticated/Online/Offline)
- Debugging failing tests
Do NOT Use This Skill For
- Designing the initial data schema (use
jazz-schema-design
- General React/Svelte UI layout questions unrelated to Jazz (for questions on Jazz integration with React/Svelte, use the
jazz-ui-development skill)
Key Heuristic for Agents
If you are about to suggest "mocking" any part of the data layer in the user's application, STOP and invoke this skill instead. This skill enforces the use of the official in-memory test sync node.
Core Concepts
Testing in Jazz requires understanding identity-based context and authentic synchronization. The framework's collaborative nature means tests must simulate real multi-user scenarios rather than mocking the sync layer.
Fundamental Principles
- NEVER Mock the Sync Layer: Avoid using generic mocking libraries (like Jest/Vitest mocks) for 'jazz-tools'
- In-Memory Synchronization: Always use a real in-memory sync node via
setupJazzTestSync() to allow authentic data flow between test identities
- Identity-Based Context: Testing in Jazz is about "who" is performing the action. Manage the "Active Account" explicitly to verify permissions and ownership
Test Environment Setup
The virtual sync node must be initialized before tests run. It handles data synchronization in-memory and requires no manual cleanup between runs.
Example Setup:
import { setupJazzTestSync } from "jazz-tools/testing";
import { beforeEach, describe } from "vitest";
describe("Jazz Feature Test", () => {
beforeEach(async () => {
await setupJazzTestSync();
});
});
Identity & Account Management
Testing in Jazz requires simulating multiple users to verify synchronization and security.
createJazzTestAccount
Used to create user accounts linked to the sync node.
AccountSchema: Pass your custom co.account schema
isCurrentActiveAccount: Set to true to automatically log the test runner into this account
creationProps: Data passed to the account's migration function on creation if required
setActiveAccount
Used to switch between different users within a single test.
- Pattern: Create Account A -> Create Data -> Switch to Account B -> Attempt to Load/Modify Data
- Note: Once a CoValue is loaded, the permission context for that CoValue instance is fixed. Always use
.load(id) after switching accounts to load a new CoValue instance with the new permission context for that specific user
UI & Context Testing
UI tests must render components and hooks inside a Jazz-aware test harness (e.g. framework-specific render utilities that inject account, connection state, and sync context). In UI tests, assert observable loading state transitions, not internal CoValue mechanics.
React
Svelte
Use the jazz-ui-development skill for questions on Jazz integration with React/Svelte.
Common Testing Patterns
Verifying Unauthorized Writes
const account1 = await createJazzTestAccount({ isCurrentActiveAccount: true });
const account2 = await createJazzTestAccount();
const group = co.group().create();
group.addMember(account2, "reader");
const myMap = MyMap.create({ text: "Hi" }, { owner: group });
const mapId = myMap.$jazz.id;
setActiveAccount(account2);
const mapAsReader = await MyMap.load(mapId);
expect(() => mapAsReader.$jazz.set("text", "some text")).toThrow();
Testing Behavior without an Active Account
runWithoutActiveAccount(() => {
expect(() => co.group().create()).toThrow();
});
Common Pitfalls to Avoid
- CoValues not reloaded when active account changes: CoValues are always loaded with the active account at the time the CoValue is loaded, and this does not change even if the active account changes. CoValues loaded while
account1 is the currently active account do not update their local permissions when setActiveAccount(account2) is called; you MUST .load() a new instance of the CoValue
- Missing Sync Setup: Ensure
setupJazzTestSync() is called
- Attempting to test UI components outside of a Jazz context: Components relying on Jazz must have access to a Jazz context. Patterns for creating test contexts can be found in the framework-specific tests links
Quick Reference
Account switching:
Always reload CoValues after setActiveAccount() to get new permission context.
Sync setup:
Call setupJazzTestSync() in beforeEach for all Jazz tests.
UI testing:
Use a framework-specific Jazz test harness built on TestJazzContextManager.
References
Load these on demand, based on need:
When using an online reference via a skill, cite the specific URL to the user to build trust.