Table of Contents
Most AssemblyScript testing tools are tied to a single runtime, usually Node.js. This works for development, but it doesn’t reflect how your code runs in production. If you deploy to WASI, Wazero, or a custom runtime, you often end up mocking everything and maintaining parallel logic just for tests. as-test solves this by letting you run tests on your actual target runtime, while only mocking what’s necessary.
Key benefits
- Runtime-agnostic: test on WASI, bindings, or custom runners
- Minimal mocking: keep real imports when possible
- Production-like testing: catch runtime-specific issues early
- Inline mocking and snapshots
- Custom reporters and coverage
- Integrated fuzzing support
The easiest way to start is with the project initializer:
npx as-test initThat gives you a basic config file, a sample test, and optionally a sample fuzzer.
If you already have a project and just want the package:
npm install --save-dev as-testBy default, as-test looks for:
- tests in
assembly/__tests__ - fuzzers in
assembly/__fuzz__ - config in
as-test.config.json
Generated files go into .as-test/.
Tests usually live in assembly/__tests__/*.spec.ts.
Example:
import { describe, expect, test } from "as-test";
describe("math", () => {
test("adds numbers", () => {
expect(1 + 2).toBe(3);
});
});Run everything:
npx ast testRun one matching file:
npx ast test mathYou do not need to learn every CLI flag to get started. Most projects can begin with npx ast test, then add more configuration only when they need it.
Mocking is supported, but the idea is to use it sparingly.
With as-test, the ideal path is to run your code against the real runtime and real imports when you can. When that is not practical, you can mock individual imports instead of rebuilding your whole environment around fake behavior.
For local functions, use mockFn and unmockFn. For host imports, use mockImport and unmockImport.
That is especially useful when:
- an import talks to the outside world
- a host function is hard to reproduce in a test
- you want to force an edge case that is difficult to trigger naturally
This keeps tests focused. You can still verify the logic in your AssemblyScript code without needing every runtime dependency to be real in every test. It also pairs well with snapshots when you want to capture the output of a mocked import and make sure it stays stable over time.
Example:
import { describe, expect, mockFn, test, unmockFn } from "as-test";
function getConfig(): string {
return "name=prod\nmode=live";
}
mockFn(getConfig, (): string => "name=demo\nmode=test");
describe("config", () => {
test("reads mocked data", () => {
expect(getConfig()).toContain("demo");
});
});
unmockFn(getConfig);For import mocking, the same idea applies, but it is usually easier to keep the imported function in a small wrapper module and mock that import path from the spec.
Snapshots are useful when the output matters more than the exact step-by-step assertions.
They work well for:
- generated strings or structured text
- serialized values
- the output of mocked imports
- larger results that would be awkward to check field by field
That lets you keep tests readable while still locking down behavior that should not change unexpectedly.
Example:
import { describe, expect, test } from "as-test";
function renderReport(): string {
return "name=demo\nmode=test";
}
describe("report", () => {
test("matches the saved output", () => {
expect(renderReport()).toMatchSnapshot();
});
});The first time you run a snapshot test, create the snapshot with:
npx ast test --update-snapshotsAfter that, a normal npx ast test will verify it.
Fuzzers usually live in assembly/__fuzz__/*.fuzz.ts.
Example:
import { expect, fuzz, FuzzSeed } from "as-test";
fuzz("bounded integer addition", (left: i32, right: i32): bool => {
const sum = left + right;
expect(sum - right).toBe(left);
return sum >= i32.MIN_VALUE;
}).generate((seed: FuzzSeed, run: (left: i32, right: i32) => bool): void => {
run(
seed.i32({ min: -1000, max: 1000 }),
seed.i32({ min: -1000, max: 1000 }),
);
});If you used npx as-test init with a fuzzer example, the config is already there. Otherwise, add a fuzz block to as-test.config.json so npx ast fuzz knows what to build:
{
"fuzz": {
"input": ["./assembly/__fuzz__/*.fuzz.ts"],
"target": "bindings"
}
}Run only fuzzers:
npx ast fuzzRun tests and fuzzers together:
npx ast test --fuzzFuzzing is there when you want broader input coverage, but it does not get in the way of the normal test flow. You can start with ordinary specs and add fuzzers later.
One of the main reasons to use as-test is that you are not locked into a single runtime.
If your project runs under WASI, bindings, or a custom runner, you can point your tests at that environment instead of treating Node.js as the only way to execute them.
For example, a simple WASI setup in as-test.config.json can look like this:
{
"input": ["./assembly/__tests__/*.spec.ts"],
"buildOptions": {
"target": "wasi"
},
"runOptions": {
"runtime": {
"cmd": "node ./.as-test/runners/default.wasi.js <file>"
}
}
}Then run your tests normally:
npx ast testIf you want to keep more than one runtime around, use modes:
{
"input": ["./assembly/__tests__/*.spec.ts"],
"modes": {
"wasi": {
"buildOptions": {
"target": "wasi"
},
"runOptions": {
"runtime": {
"cmd": "node ./.as-test/runners/default.wasi.js <file>"
}
}
},
"bindings": {
"buildOptions": {
"target": "bindings"
},
"runOptions": {
"runtime": {
"cmd": "node ./.as-test/runners/default.bindings.js <file>"
}
}
}
}
}Run a specific mode with:
npx ast test --mode wasior
npx ast test --mode wasi,bindingsThis is the general idea throughout the project: write tests once, then choose the runtime that matches how your code actually runs.
Runnable example projects live in examples/. They are useful if you want to see complete setups instead of isolated snippets.
This project is distributed under an open source license. Work on this project is done by passion, but if you want to support it financially, you can do so by making a donation to the project's GitHub Sponsors page.
You can view the full license using the following link: License
Please send all issues to GitHub Issues and to converse, please send me an email at me@jairus.dev
- Email: Send me inquiries, questions, or requests at me@jairus.dev
- GitHub: Visit the official GitHub repository Here
- Website: Visit my official website at jairus.dev
- Discord: Contact me at My Discord or on the AssemblyScript Discord Server