mirror of
https://github.com/openclaw/openclaw.git
synced 2026-04-10 08:41:13 +00:00
* feat(gateway): add skills.search and skills.detail RPC methods Expose ClawHub search and detail capabilities through the Gateway protocol, enabling desktop/web clients to browse and inspect skills from the registry. New RPCs: - skills.search: search ClawHub skills by query with optional limit - skills.detail: fetch full detail for a single skill by slug Both methods delegate to existing agent-layer functions (searchSkillsFromClawHub, fetchSkillDetailFromClawHub) which wrap the ClawHub HTTP client. No new external dependencies. Signed-off-by: samzong <samzong.lu@gmail.com> * feat(skills): add ClawHub skill search and detail in Control UI Add skills.search and skills.detail Gateway RPC methods with typed protocol schemas, AJV validators, and handler implementations. Wire the new RPCs into the Control UI Skills panel with a debounced search input, results list, detail dialog, and one-click install from ClawHub. Gateway: - SkillsSearchParams/ResultSchema and SkillsDetailParams/ResultSchema - Handler calls searchClawHubSkills and fetchClawHubSkillDetail directly - Remove zero-logic fetchSkillDetailFromClawHub wrapper - 9 handler tests including boundary validation Control UI: - searchClawHub, loadClawHubDetail, installFromClawHub controllers - 300ms debounced search input to avoid 429 rate limits - Dedicated install busy state (clawhubInstallSlug) with success/error feedback - Install buttons disabled during install with progress text - Detail dialog with owner, version, changelog, platform metadata Part of #43301 Signed-off-by: samzong <samzong.lu@gmail.com> * fix(skills): guard search and detail responses against stale writes Signed-off-by: samzong <samzong.lu@gmail.com> * fix(skills): reset loading flags on query clear and detail close Signed-off-by: samzong <samzong.lu@gmail.com> * fix(gateway): register skills.search/detail in read scope and method list Add skills.search and skills.detail to the operator READ scope group and the server methods list. Without this, unclassified methods default to operator.admin, blocking read-only operator sessions. Also guard the detail loading reset in the finally block by the active slug to prevent a transient flash when rapidly switching skills. Signed-off-by: samzong <samzong.lu@gmail.com> * fix(skills): guard search loading reset by active query Signed-off-by: samzong <samzong.lu@gmail.com> * test: cover ClawHub skills UI flow * fix: clear stale ClawHub search results --------- Signed-off-by: samzong <samzong.lu@gmail.com> Co-authored-by: Frank Yang <frank.ekn@gmail.com>
209 lines
7.2 KiB
TypeScript
209 lines
7.2 KiB
TypeScript
import path from "node:path";
|
|
import { fileURLToPath } from "node:url";
|
|
import { defineConfig } from "vitest/config";
|
|
import {
|
|
BUNDLED_PLUGIN_ROOT_DIR,
|
|
BUNDLED_PLUGIN_TEST_GLOB,
|
|
} from "./scripts/lib/bundled-plugin-paths.mjs";
|
|
import { pluginSdkSubpaths } from "./scripts/lib/plugin-sdk-entries.mjs";
|
|
import { resolveLocalVitestMaxWorkers } from "./scripts/test-planner/runtime-profile.mjs";
|
|
import {
|
|
behaviorManifestPath,
|
|
unitMemoryHotspotManifestPath,
|
|
unitTimingManifestPath,
|
|
} from "./scripts/test-runner-manifest.mjs";
|
|
import { loadVitestExperimentalConfig } from "./vitest.performance-config.ts";
|
|
|
|
export { resolveLocalVitestMaxWorkers };
|
|
|
|
const repoRoot = path.dirname(fileURLToPath(import.meta.url));
|
|
const isCI = process.env.CI === "true" || process.env.GITHUB_ACTIONS === "true";
|
|
const isWindows = process.platform === "win32";
|
|
const localWorkers = resolveLocalVitestMaxWorkers();
|
|
const ciWorkers = isWindows ? 2 : 3;
|
|
export default defineConfig({
|
|
resolve: {
|
|
// Keep this ordered: the base `openclaw/plugin-sdk` alias is a prefix match.
|
|
alias: [
|
|
{
|
|
find: "openclaw/extension-api",
|
|
replacement: path.join(repoRoot, "src", "extensionAPI.ts"),
|
|
},
|
|
...pluginSdkSubpaths.map((subpath) => ({
|
|
find: `openclaw/plugin-sdk/${subpath}`,
|
|
replacement: path.join(repoRoot, "src", "plugin-sdk", `${subpath}.ts`),
|
|
})),
|
|
{
|
|
find: "openclaw/plugin-sdk",
|
|
replacement: path.join(repoRoot, "src", "plugin-sdk", "index.ts"),
|
|
},
|
|
],
|
|
},
|
|
test: {
|
|
testTimeout: 120_000,
|
|
hookTimeout: isWindows ? 180_000 : 120_000,
|
|
// Many suites rely on `vi.stubEnv(...)` and expect it to be scoped to the test.
|
|
// Keep env restoration automatic so shared-worker runs do not leak state.
|
|
unstubEnvs: true,
|
|
// Same rationale as unstubEnvs: avoid cross-test pollution from shared globals.
|
|
unstubGlobals: true,
|
|
pool: "forks",
|
|
maxWorkers: isCI ? ciWorkers : localWorkers,
|
|
forceRerunTriggers: [
|
|
"package.json",
|
|
"pnpm-lock.yaml",
|
|
"test/setup.ts",
|
|
"test/setup.shared.ts",
|
|
"test/setup.extensions.ts",
|
|
"scripts/test-parallel.mjs",
|
|
"scripts/test-planner/catalog.mjs",
|
|
"scripts/test-planner/executor.mjs",
|
|
"scripts/test-planner/planner.mjs",
|
|
"scripts/test-planner/runtime-profile.mjs",
|
|
"scripts/test-runner-manifest.mjs",
|
|
"vitest.channel-paths.mjs",
|
|
"vitest.channels.config.ts",
|
|
"vitest.bundled.config.ts",
|
|
"vitest.config.ts",
|
|
"vitest.contracts.config.ts",
|
|
"vitest.e2e.config.ts",
|
|
"vitest.extensions.config.ts",
|
|
"vitest.gateway.config.ts",
|
|
"vitest.live.config.ts",
|
|
"vitest.performance-config.ts",
|
|
"vitest.scoped-config.ts",
|
|
"vitest.unit.config.ts",
|
|
"vitest.unit-paths.mjs",
|
|
behaviorManifestPath,
|
|
unitTimingManifestPath,
|
|
unitMemoryHotspotManifestPath,
|
|
],
|
|
include: [
|
|
"src/**/*.test.ts",
|
|
BUNDLED_PLUGIN_TEST_GLOB,
|
|
"packages/**/*.test.ts",
|
|
"test/**/*.test.ts",
|
|
"ui/src/ui/app-chat.test.ts",
|
|
"ui/src/ui/chat/**/*.test.ts",
|
|
"ui/src/ui/views/agents-utils.test.ts",
|
|
"ui/src/ui/views/channels.test.ts",
|
|
"ui/src/ui/views/chat.test.ts",
|
|
"ui/src/ui/views/nodes.devices.test.ts",
|
|
"ui/src/ui/views/skills.test.ts",
|
|
"ui/src/ui/views/usage-render-details.test.ts",
|
|
"ui/src/ui/controllers/agents.test.ts",
|
|
"ui/src/ui/controllers/chat.test.ts",
|
|
"ui/src/ui/controllers/skills.test.ts",
|
|
"ui/src/ui/controllers/sessions.test.ts",
|
|
"ui/src/ui/views/sessions.test.ts",
|
|
"ui/src/ui/app-tool-stream.node.test.ts",
|
|
"ui/src/ui/app-gateway.sessions.node.test.ts",
|
|
"ui/src/ui/chat/slash-command-executor.node.test.ts",
|
|
],
|
|
setupFiles: ["test/setup.ts"],
|
|
exclude: [
|
|
"dist/**",
|
|
"test/fixtures/**",
|
|
"apps/macos/**",
|
|
"apps/macos/.build/**",
|
|
"**/node_modules/**",
|
|
"**/vendor/**",
|
|
"dist/OpenClaw.app/**",
|
|
"**/*.live.test.ts",
|
|
"**/*.e2e.test.ts",
|
|
],
|
|
coverage: {
|
|
provider: "v8",
|
|
reporter: ["text", "lcov"],
|
|
// Keep coverage stable without an ever-growing exclude list:
|
|
// only count files actually exercised by the test suite.
|
|
all: false,
|
|
thresholds: {
|
|
lines: 70,
|
|
functions: 70,
|
|
branches: 55,
|
|
statements: 70,
|
|
},
|
|
// Anchor to repo-root `src/` only. Without this, coverage globs can
|
|
// unintentionally match nested `*/src/**` folders (extensions, apps, etc).
|
|
include: ["./src/**/*.ts"],
|
|
exclude: [
|
|
// Never count workspace packages/apps toward core coverage thresholds.
|
|
`${BUNDLED_PLUGIN_ROOT_DIR}/**`,
|
|
"apps/**",
|
|
"ui/**",
|
|
"test/**",
|
|
"src/**/*.test.ts",
|
|
// Entrypoints and wiring (covered by CI smoke + manual/e2e flows).
|
|
"src/entry.ts",
|
|
"src/index.ts",
|
|
"src/runtime.ts",
|
|
"src/channel-web.ts",
|
|
"src/logging.ts",
|
|
"src/cli/**",
|
|
"src/commands/**",
|
|
"src/daemon/**",
|
|
"src/hooks/**",
|
|
"src/macos/**",
|
|
|
|
// Large integration surfaces; validated via e2e/manual/contract tests.
|
|
"src/acp/**",
|
|
"src/agents/**",
|
|
"src/channels/**",
|
|
"src/gateway/**",
|
|
"src/line/**",
|
|
"src/media-understanding/**",
|
|
"src/node-host/**",
|
|
"src/plugins/**",
|
|
"src/providers/**",
|
|
|
|
// Some agent integrations are intentionally validated via manual/e2e runs.
|
|
"src/agents/model-scan.ts",
|
|
"src/agents/pi-embedded-runner.ts",
|
|
"src/agents/sandbox-paths.ts",
|
|
"src/agents/sandbox.ts",
|
|
"src/agents/skills-install.ts",
|
|
"src/agents/pi-tool-definition-adapter.ts",
|
|
"src/agents/tools/discord-actions*.ts",
|
|
"src/agents/tools/slack-actions.ts",
|
|
|
|
// Hard-to-unit-test modules; exercised indirectly by integration tests.
|
|
"src/infra/state-migrations.ts",
|
|
"src/infra/skills-remote.ts",
|
|
"src/infra/update-check.ts",
|
|
"src/infra/ports-inspect.ts",
|
|
"src/infra/outbound/outbound-session.ts",
|
|
"src/memory/batch-gemini.ts",
|
|
|
|
// Gateway server integration surfaces are intentionally validated via manual/e2e runs.
|
|
"src/gateway/control-ui.ts",
|
|
"src/gateway/server-bridge.ts",
|
|
"src/gateway/server-channels.ts",
|
|
"src/gateway/server-methods/config.ts",
|
|
"src/gateway/server-methods/send.ts",
|
|
"src/gateway/server-methods/skills.ts",
|
|
"src/gateway/server-methods/talk.ts",
|
|
"src/gateway/server-methods/web.ts",
|
|
"src/gateway/server-methods/wizard.ts",
|
|
|
|
// Process bridges are hard to unit-test in isolation.
|
|
"src/gateway/call.ts",
|
|
"src/process/tau-rpc.ts",
|
|
"src/process/exec.ts",
|
|
// Interactive UIs/flows are intentionally validated via manual/e2e runs.
|
|
"src/tui/**",
|
|
"src/wizard/**",
|
|
// Channel surfaces are largely integration-tested (or manually validated).
|
|
"src/browser/**",
|
|
"src/channels/web/**",
|
|
"src/webchat/**",
|
|
"src/gateway/server.ts",
|
|
"src/gateway/client.ts",
|
|
"src/gateway/protocol/**",
|
|
"src/infra/tailscale.ts",
|
|
],
|
|
},
|
|
...loadVitestExperimentalConfig(),
|
|
},
|
|
});
|