18 KiB
summary, title, sidebarTitle, read_when
| summary | title | sidebarTitle | read_when | |||
|---|---|---|---|---|---|---|
| Testing utilities and patterns for OpenClaw plugins | Plugin testing | Testing |
|
Reference for test utilities, patterns, and lint enforcement for OpenClaw plugins.
**Looking for test examples?** The how-to guides include worked test examples: [Channel plugin tests](/plugins/sdk-channel-plugins#step-6-test) and [Provider plugin tests](/plugins/sdk-provider-plugins#step-6-test).Test utilities
Compatibility import: openclaw/plugin-sdk/testing
Plugin API mock import: openclaw/plugin-sdk/plugin-test-api
Channel contract import: openclaw/plugin-sdk/channel-contract-testing
Channel test helper import: openclaw/plugin-sdk/channel-test-helpers
Channel target test import: openclaw/plugin-sdk/channel-target-testing
Plugin contract import: openclaw/plugin-sdk/plugin-test-contracts
Plugin runtime test import: openclaw/plugin-sdk/plugin-test-runtime
Provider contract import: openclaw/plugin-sdk/provider-test-contracts
Environment/network test import: openclaw/plugin-sdk/test-env
Generic fixture import: openclaw/plugin-sdk/test-fixtures
Prefer the focused subpaths below for new plugin tests. The broad
openclaw/plugin-sdk/testing barrel remains for compatibility with older tests
and helpers that have not moved to a narrower documented surface yet.
import {
shouldAckReaction,
removeAckReactionAfterReply,
} from "openclaw/plugin-sdk/channel-feedback";
import { installCommonResolveTargetErrorCases } from "openclaw/plugin-sdk/channel-target-testing";
import { createTestPluginApi } from "openclaw/plugin-sdk/plugin-test-api";
import { expectChannelInboundContextContract } from "openclaw/plugin-sdk/channel-contract-testing";
import { createStartAccountContext } from "openclaw/plugin-sdk/channel-test-helpers";
import { describePluginRegistrationContract } from "openclaw/plugin-sdk/plugin-test-contracts";
import { registerSingleProviderPlugin } from "openclaw/plugin-sdk/plugin-test-runtime";
import { describeOpenAIProviderRuntimeContract } from "openclaw/plugin-sdk/provider-test-contracts";
import { withEnv, withFetchPreconnect } from "openclaw/plugin-sdk/test-env";
import { createCliRuntimeCapture, typedCases } from "openclaw/plugin-sdk/test-fixtures";
Available exports
| Export | Purpose |
|---|---|
createTestPluginApi |
Build a minimal plugin API mock for direct registration unit tests. Import from plugin-sdk/plugin-test-api |
expectChannelInboundContextContract |
Assert channel inbound context shape. Import from plugin-sdk/channel-contract-testing |
installChannelOutboundPayloadContractSuite |
Install channel outbound payload contract cases. Import from plugin-sdk/channel-contract-testing |
createStartAccountContext |
Build channel account lifecycle contexts. Import from plugin-sdk/channel-test-helpers |
installChannelActionsContractSuite |
Install generic channel message-action contract cases. Import from plugin-sdk/channel-test-helpers |
installChannelSetupContractSuite |
Install generic channel setup contract cases. Import from plugin-sdk/channel-test-helpers |
installChannelStatusContractSuite |
Install generic channel status contract cases. Import from plugin-sdk/channel-test-helpers |
expectDirectoryIds |
Assert channel directory ids from a directory-list function. Import from plugin-sdk/channel-test-helpers |
describePluginRegistrationContract |
Install plugin registration contract checks. Import from plugin-sdk/plugin-test-contracts |
registerSingleProviderPlugin |
Register one provider plugin in loader smoke tests. Import from plugin-sdk/plugin-test-runtime |
registerProviderPlugin |
Capture all provider kinds from one plugin. Import from plugin-sdk/plugin-test-runtime |
registerProviderPlugins |
Capture provider registrations across multiple plugins. Import from plugin-sdk/plugin-test-runtime |
requireRegisteredProvider |
Assert that a provider collection contains an id. Import from plugin-sdk/plugin-test-runtime |
createRuntimeEnv |
Build a mocked CLI/plugin runtime environment. Import from plugin-sdk/plugin-test-runtime |
createPluginSetupWizardStatus |
Build setup status helpers for channel plugins. Import from plugin-sdk/plugin-test-runtime |
describeOpenAIProviderRuntimeContract |
Install provider-family runtime contract checks. Import from plugin-sdk/provider-test-contracts |
installCommonResolveTargetErrorCases |
Shared test cases for target resolution error handling. Import from plugin-sdk/channel-target-testing |
shouldAckReaction |
Check whether a channel should add an ack reaction. Import from plugin-sdk/channel-feedback |
removeAckReactionAfterReply |
Remove ack reaction after reply delivery. Import from plugin-sdk/channel-feedback |
createTestRegistry |
Build a channel plugin registry fixture. Import from plugin-sdk/plugin-test-runtime or plugin-sdk/channel-test-helpers |
createEmptyPluginRegistry |
Build an empty plugin registry fixture. Import from plugin-sdk/plugin-test-runtime or plugin-sdk/channel-test-helpers |
setActivePluginRegistry |
Install a registry fixture for plugin runtime tests. Import from plugin-sdk/plugin-test-runtime or plugin-sdk/channel-test-helpers |
createRequestCaptureJsonFetch |
Capture JSON fetch requests in media helper tests. Import from plugin-sdk/test-env |
withFetchPreconnect |
Run fetch tests with preconnect hooks installed. Import from plugin-sdk/test-env |
withEnv / withEnvAsync |
Temporarily patch environment variables. Import from plugin-sdk/test-env |
createTempHomeEnv / withTempDir |
Create isolated filesystem test fixtures. Import from plugin-sdk/test-env |
createMockServerResponse |
Create a minimal HTTP server response mock. Import from plugin-sdk/test-env |
createCliRuntimeCapture |
Capture CLI runtime output in tests. Import from plugin-sdk/test-fixtures |
createSandboxTestContext |
Build sandbox test contexts. Import from plugin-sdk/test-fixtures |
writeSkill |
Write skill fixtures. Import from plugin-sdk/test-fixtures |
makeAgentAssistantMessage |
Build agent transcript message fixtures. Import from plugin-sdk/test-fixtures |
peekSystemEvents / resetSystemEventsForTest |
Inspect and reset system event fixtures. Import from plugin-sdk/test-fixtures |
sanitizeTerminalText |
Sanitize terminal output for assertions. Import from plugin-sdk/test-fixtures |
countLines / hasBalancedFences |
Assert chunking output shape. Import from plugin-sdk/test-fixtures |
runProviderCatalog |
Execute a provider catalog hook with test dependencies |
resolveProviderWizardOptions |
Resolve provider setup wizard choices in contract tests |
resolveProviderModelPickerEntries |
Resolve provider model-picker entries in contract tests |
buildProviderPluginMethodChoice |
Build provider wizard choice ids for assertions |
setProviderWizardProvidersResolverForTest |
Inject provider wizard providers for isolated tests |
createProviderUsageFetch |
Build provider usage fetch fixtures |
useFrozenTime / useRealTime |
Freeze and restore timers for time-sensitive tests. Import from plugin-sdk/test-env |
createTestWizardPrompter |
Build a mocked setup wizard prompter |
createRuntimeTaskFlow |
Create isolated runtime task-flow state |
typedCases |
Preserve literal types for table-driven tests. Import from plugin-sdk/test-fixtures |
Bundled-plugin contract suites also use SDK testing subpaths for test-only
registry, manifest, public-artifact, and runtime fixture helpers. Core-only
suites that depend on bundled OpenClaw inventory stay under src/plugins/contracts.
Keep new extension tests on a documented focused SDK subpath such as
plugin-sdk/plugin-test-api, plugin-sdk/channel-contract-testing,
plugin-sdk/channel-test-helpers, plugin-sdk/plugin-test-contracts,
plugin-sdk/plugin-test-runtime, plugin-sdk/provider-test-contracts,
plugin-sdk/test-env, or plugin-sdk/test-fixtures rather than importing the
broad plugin-sdk/testing compatibility barrel, repo src/** files, or repo
test/helpers/plugins/* bridges directly.
Types
The testing subpath also re-exports types useful in test files:
import type {
ChannelAccountSnapshot,
ChannelGatewayContext,
OpenClawConfig,
PluginRuntime,
RuntimeEnv,
MockFn,
} from "openclaw/plugin-sdk/testing";
Testing target resolution
Use installCommonResolveTargetErrorCases to add standard error cases for
channel target resolution:
import { describe } from "vitest";
import { installCommonResolveTargetErrorCases } from "openclaw/plugin-sdk/channel-target-testing";
describe("my-channel target resolution", () => {
installCommonResolveTargetErrorCases({
resolveTarget: ({ to, mode, allowFrom }) => {
// Your channel's target resolution logic
return myChannelResolveTarget({ to, mode, allowFrom });
},
implicitAllowFrom: ["user1", "user2"],
});
// Add channel-specific test cases
it("should resolve @username targets", () => {
// ...
});
});
Testing patterns
Testing registration contracts
Unit tests that pass a hand-written api mock to register(api) do not exercise
OpenClaw's loader acceptance gates. Add at least one loader-backed smoke test
for each registration surface your plugin depends on, especially hooks and
exclusive capabilities such as memory.
The real loader fails plugin registration when required metadata is missing or a
plugin calls a capability API it does not own. For example,
api.registerHook(...) requires a hook name, and
api.registerMemoryCapability(...) requires the plugin manifest or exported
entry to declare kind: "memory".
Testing runtime config access
Prefer the shared plugin runtime mock from openclaw/plugin-sdk/channel-test-helpers
when testing bundled channel plugins. Its deprecated runtime.config.loadConfig() and
runtime.config.writeConfigFile(...) mocks throw by default so tests catch new
usage of compatibility APIs. Override those mocks only when the test is
explicitly covering legacy compatibility behavior.
Unit testing a channel plugin
import { describe, it, expect, vi } from "vitest";
describe("my-channel plugin", () => {
it("should resolve account from config", () => {
const cfg = {
channels: {
"my-channel": {
token: "test-token",
allowFrom: ["user1"],
},
},
};
const account = myPlugin.setup.resolveAccount(cfg, undefined);
expect(account.token).toBe("test-token");
});
it("should inspect account without materializing secrets", () => {
const cfg = {
channels: {
"my-channel": { token: "test-token" },
},
};
const inspection = myPlugin.setup.inspectAccount(cfg, undefined);
expect(inspection.configured).toBe(true);
expect(inspection.tokenStatus).toBe("available");
// No token value exposed
expect(inspection).not.toHaveProperty("token");
});
});
Unit testing a provider plugin
import { describe, it, expect } from "vitest";
describe("my-provider plugin", () => {
it("should resolve dynamic models", () => {
const model = myProvider.resolveDynamicModel({
modelId: "custom-model-v2",
// ... context
});
expect(model.id).toBe("custom-model-v2");
expect(model.provider).toBe("my-provider");
expect(model.api).toBe("openai-completions");
});
it("should return catalog when API key is available", async () => {
const result = await myProvider.catalog.run({
resolveProviderApiKey: () => ({ apiKey: "test-key" }),
// ... context
});
expect(result?.provider?.models).toHaveLength(2);
});
});
Mocking the plugin runtime
For code that uses createPluginRuntimeStore, mock the runtime in tests:
import { createPluginRuntimeStore } from "openclaw/plugin-sdk/runtime-store";
import type { PluginRuntime } from "openclaw/plugin-sdk/runtime-store";
const store = createPluginRuntimeStore<PluginRuntime>({
pluginId: "test-plugin",
errorMessage: "test runtime not set",
});
// In test setup
const mockRuntime = {
agent: {
resolveAgentDir: vi.fn().mockReturnValue("/tmp/agent"),
// ... other mocks
},
config: {
current: vi.fn(() => ({}) as const),
mutateConfigFile: vi.fn(),
replaceConfigFile: vi.fn(),
},
// ... other namespaces
} as unknown as PluginRuntime;
store.setRuntime(mockRuntime);
// After tests
store.clearRuntime();
Testing with per-instance stubs
Prefer per-instance stubs over prototype mutation:
// Preferred: per-instance stub
const client = new MyChannelClient();
client.sendMessage = vi.fn().mockResolvedValue({ id: "msg-1" });
// Avoid: prototype mutation
// MyChannelClient.prototype.sendMessage = vi.fn();
Contract tests (in-repo plugins)
Bundled plugins have contract tests that verify registration ownership:
pnpm test -- src/plugins/contracts/
These tests assert:
- Which plugins register which providers
- Which plugins register which speech providers
- Registration shape correctness
- Runtime contract compliance
Running scoped tests
For a specific plugin:
pnpm test -- <bundled-plugin-root>/my-channel/
For contract tests only:
pnpm test -- src/plugins/contracts/shape.contract.test.ts
pnpm test -- src/plugins/contracts/auth.contract.test.ts
pnpm test -- src/plugins/contracts/runtime.contract.test.ts
Lint enforcement (in-repo plugins)
Three rules are enforced by pnpm check for in-repo plugins:
- No monolithic root imports --
openclaw/plugin-sdkroot barrel is rejected - No direct
src/imports -- plugins cannot import../../src/directly - No self-imports -- plugins cannot import their own
plugin-sdk/<name>subpath
External plugins are not subject to these lint rules, but following the same patterns is recommended.
Test configuration
OpenClaw uses Vitest with V8 coverage thresholds. For plugin tests:
# Run all tests
pnpm test
# Run specific plugin tests
pnpm test -- <bundled-plugin-root>/my-channel/src/channel.test.ts
# Run with a specific test name filter
pnpm test -- <bundled-plugin-root>/my-channel/ -t "resolves account"
# Run with coverage
pnpm test:coverage
If local runs cause memory pressure:
OPENCLAW_VITEST_MAX_WORKERS=1 pnpm test
Related
- SDK Overview -- import conventions
- SDK Channel Plugins -- channel plugin interface
- SDK Provider Plugins -- provider plugin hooks
- Building Plugins -- getting started guide