From 28b8046ed0a1e44e85c8f9ea90ed733ee910d2b1 Mon Sep 17 00:00:00 2001 From: Gustavo Madeira Santana Date: Thu, 16 Apr 2026 22:45:32 -0400 Subject: [PATCH] test(matrix): cover subagent hook registration --- CHANGELOG.md | 1 + extensions/matrix/index.test.ts | 42 +++++++++++++++++++++++++++++++++ 2 files changed, 43 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 66cdaf15bb6..2dbc786fe64 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -58,6 +58,7 @@ Docs: https://docs.openclaw.ai - Ollama/chat: strip the `ollama/` provider prefix from Ollama chat request model ids so configured refs like `ollama/qwen3:14b-q8_0` stop 404ing against the Ollama API. (#67457) Thanks @suboss87. - Agents/tools: resolve non-workspace host tilde paths against the OS home directory and keep edit recovery aligned with that same path target, so `~/...` host edit/write operations stop failing or reading back the wrong file when `OPENCLAW_HOME` differs. (#62804) Thanks @stainlu. - Speech/TTS: auto-enable the bundled Microsoft and ElevenLabs speech providers, and route generic TTS directive tokens through the explicit or active provider first so overrides like `[[tts:speed=1.2]]` stop silently landing on the wrong provider. (#62846) Thanks @stainlu. +<<<<<<< HEAD - OpenAI Codex/models: normalize stale native transport metadata in both runtime resolution and discovery/listing so legacy `openai-codex` rows with missing `api` or `https://chatgpt.com/backend-api/v1` self-heal to the canonical Codex transport instead of routing requests through broken HTML/Cloudflare paths, combining the original fixes proposed in #66969 (saamuelng601-pixel) and #67159 (hclsys). (#67635) - Agents/failover: treat HTML provider error pages as upstream transport failures for CDN-style 5xx responses without misclassifying embedded body text as API rate limits, while still preserving auth remediation for HTML 401/403 pages and proxy remediation for HTML 407 pages. (#67642) Thanks @stainlu. - Gateway/skills: bump the cached skills-snapshot version whenever a config write touches `skills.*` (for example `skills.allowBundled`, `skills.entries..enabled`, or `skills.profile`). Existing agent sessions persist a `skillsSnapshot` in `sessions.json` that reuses the skill list frozen at session creation; without this invalidation, removing a bundled skill from the allowlist left the old snapshot live and the model kept calling the disabled tool, producing `Tool not found` loops that ran until the embedded-run timeout. (#67401) Thanks @xantorres. diff --git a/extensions/matrix/index.test.ts b/extensions/matrix/index.test.ts index 6f2dcddcc9f..60c42c1c257 100644 --- a/extensions/matrix/index.test.ts +++ b/extensions/matrix/index.test.ts @@ -7,12 +7,23 @@ const cliMocks = vi.hoisted(() => ({ registerMatrixCli: vi.fn(), })); +const runtimeMocks = vi.hoisted(() => ({ + ensureMatrixCryptoRuntime: vi.fn(async () => {}), + handleVerificationBootstrap: vi.fn(async () => {}), + handleVerificationStatus: vi.fn(async () => {}), + handleVerifyRecoveryKey: vi.fn(async () => {}), + setMatrixRuntime: vi.fn(), +})); + vi.mock("./src/cli.js", () => { return { registerMatrixCli: cliMocks.registerMatrixCli, }; }); +vi.mock("./plugin-entry.handlers.runtime.js", () => runtimeMocks); +vi.mock("./runtime-api.js", () => ({ setMatrixRuntime: runtimeMocks.setMatrixRuntime })); + describe("matrix plugin", () => { it("registers matrix CLI through a descriptor-backed lazy registrar", async () => { const registerCli = vi.fn(); @@ -56,4 +67,35 @@ describe("matrix plugin", () => { expect(entry.id).toBe("matrix"); expect(entry.name).toBe("Matrix"); }); + + it("registers subagent lifecycle hooks during full registration", () => { + const on = vi.fn(); + const registerChannel = vi.fn(); + const registerGatewayMethod = vi.fn(); + const api = createTestPluginApi({ + id: "matrix", + name: "Matrix", + source: "test", + config: {}, + runtime: {} as never, + registrationMode: "full", + on, + registerChannel, + registerGatewayMethod, + }); + + entry.register(api); + + expect(registerChannel).toHaveBeenCalledWith({ + plugin: expect.objectContaining({ id: "matrix" }), + }); + expect(on.mock.calls.map(([hookName]) => hookName)).toEqual([ + "subagent_spawning", + "subagent_ended", + "subagent_delivery_target", + ]); + for (const [, handler] of on.mock.calls) { + expect(handler).toEqual(expect.any(Function)); + } + }); });