mirror of https://github.com/openclaw/openclaw.git
406 lines
12 KiB
TypeScript
406 lines
12 KiB
TypeScript
import os from "node:os";
|
|
import path from "node:path";
|
|
import { fileURLToPath } from "node:url";
|
|
import { pluginSdkSubpaths } from "./scripts/lib/plugin-sdk-entries.mjs";
|
|
import {
|
|
BUNDLED_PLUGIN_ROOT_DIR,
|
|
BUNDLED_PLUGIN_TEST_GLOB,
|
|
} from "./vitest.bundled-plugin-paths.ts";
|
|
import { loadVitestExperimentalConfig } from "./vitest.performance-config.ts";
|
|
import {
|
|
detectVitestProcessStats,
|
|
shouldPrintVitestThrottle,
|
|
type VitestProcessStats,
|
|
} from "./vitest.system-load.ts";
|
|
|
|
const clamp = (value: number, min: number, max: number) => Math.max(min, Math.min(max, value));
|
|
|
|
function parsePositiveInt(value: string | undefined): number | null {
|
|
const parsed = Number.parseInt(value ?? "", 10);
|
|
return Number.isFinite(parsed) && parsed > 0 ? parsed : null;
|
|
}
|
|
|
|
function isSystemThrottleDisabled(env: Record<string, string | undefined>): boolean {
|
|
const normalized = env.OPENCLAW_VITEST_DISABLE_SYSTEM_THROTTLE?.trim().toLowerCase();
|
|
return normalized === "1" || normalized === "true";
|
|
}
|
|
|
|
type VitestHostInfo = {
|
|
cpuCount?: number;
|
|
loadAverage1m?: number;
|
|
totalMemoryBytes?: number;
|
|
};
|
|
|
|
export type OpenClawVitestPool = "threads";
|
|
|
|
export type LocalVitestScheduling = {
|
|
maxWorkers: number;
|
|
fileParallelism: boolean;
|
|
throttledBySystem: boolean;
|
|
};
|
|
|
|
export const jsdomOptimizedDeps = {
|
|
optimizer: {
|
|
web: {
|
|
enabled: true,
|
|
include: ["lit", "lit-html", "@lit/reactive-element", "marked"] as string[],
|
|
},
|
|
},
|
|
};
|
|
|
|
function detectVitestHostInfo(): Required<VitestHostInfo> {
|
|
return {
|
|
cpuCount:
|
|
typeof os.availableParallelism === "function" ? os.availableParallelism() : os.cpus().length,
|
|
loadAverage1m: os.loadavg()[0] ?? 0,
|
|
totalMemoryBytes: os.totalmem(),
|
|
};
|
|
}
|
|
|
|
export function resolveLocalVitestMaxWorkers(
|
|
env: Record<string, string | undefined> = process.env,
|
|
system: VitestHostInfo = detectVitestHostInfo(),
|
|
pool: OpenClawVitestPool = resolveDefaultVitestPool(env),
|
|
processStats: VitestProcessStats = detectVitestProcessStats(env),
|
|
): number {
|
|
return resolveLocalVitestScheduling(env, system, pool, processStats).maxWorkers;
|
|
}
|
|
|
|
export function resolveLocalVitestScheduling(
|
|
env: Record<string, string | undefined> = process.env,
|
|
system: VitestHostInfo = detectVitestHostInfo(),
|
|
pool: OpenClawVitestPool = resolveDefaultVitestPool(env),
|
|
processStats: VitestProcessStats = detectVitestProcessStats(env),
|
|
): LocalVitestScheduling {
|
|
const override = parsePositiveInt(env.OPENCLAW_VITEST_MAX_WORKERS ?? env.OPENCLAW_TEST_WORKERS);
|
|
if (override !== null) {
|
|
const maxWorkers = clamp(override, 1, 16);
|
|
return {
|
|
maxWorkers,
|
|
fileParallelism: maxWorkers > 1,
|
|
throttledBySystem: false,
|
|
};
|
|
}
|
|
|
|
const cpuCount = Math.max(1, system.cpuCount ?? 1);
|
|
const loadAverage1m = Math.max(0, system.loadAverage1m ?? 0);
|
|
const totalMemoryGb = (system.totalMemoryBytes ?? 0) / 1024 ** 3;
|
|
|
|
let inferred =
|
|
cpuCount <= 4 ? 1 : cpuCount <= 8 ? 2 : cpuCount <= 12 ? 3 : cpuCount <= 16 ? 4 : 6;
|
|
|
|
if (totalMemoryGb <= 16) {
|
|
inferred = Math.min(inferred, 2);
|
|
} else if (totalMemoryGb <= 32) {
|
|
inferred = Math.min(inferred, 3);
|
|
} else if (totalMemoryGb <= 64) {
|
|
inferred = Math.min(inferred, 4);
|
|
} else if (totalMemoryGb <= 128) {
|
|
inferred = Math.min(inferred, 5);
|
|
} else {
|
|
inferred = Math.min(inferred, 6);
|
|
}
|
|
|
|
const loadRatio = loadAverage1m > 0 ? loadAverage1m / cpuCount : 0;
|
|
if (loadRatio >= 1) {
|
|
inferred = Math.max(1, Math.floor(inferred / 2));
|
|
} else if (loadRatio >= 0.75) {
|
|
inferred = Math.max(1, inferred - 1);
|
|
}
|
|
|
|
if (pool === "threads") {
|
|
inferred = Math.min(inferred, 4);
|
|
if (cpuCount >= 8) {
|
|
inferred = Math.max(1, inferred - 1);
|
|
}
|
|
if (loadRatio >= 0.5) {
|
|
inferred = Math.max(1, inferred - 1);
|
|
}
|
|
}
|
|
|
|
inferred = clamp(inferred, 1, 16);
|
|
|
|
if (isSystemThrottleDisabled(env)) {
|
|
return {
|
|
maxWorkers: inferred,
|
|
fileParallelism: true,
|
|
throttledBySystem: false,
|
|
};
|
|
}
|
|
|
|
const highSystemContention =
|
|
loadRatio >= 1 ||
|
|
processStats.otherVitestWorkerCount >= 2 ||
|
|
processStats.otherVitestCpuPercent >= 150 ||
|
|
processStats.otherVitestRootCount >= 2;
|
|
|
|
if (highSystemContention) {
|
|
return {
|
|
maxWorkers: 1,
|
|
fileParallelism: false,
|
|
throttledBySystem: true,
|
|
};
|
|
}
|
|
|
|
const moderateSystemContention =
|
|
loadRatio >= 0.75 ||
|
|
processStats.otherVitestWorkerCount >= 1 ||
|
|
processStats.otherVitestCpuPercent >= 75 ||
|
|
processStats.otherVitestRootCount >= 1;
|
|
|
|
if (moderateSystemContention) {
|
|
const maxWorkers = Math.min(inferred, 2);
|
|
return {
|
|
maxWorkers,
|
|
fileParallelism: true,
|
|
throttledBySystem: maxWorkers < inferred,
|
|
};
|
|
}
|
|
|
|
return {
|
|
maxWorkers: inferred,
|
|
fileParallelism: true,
|
|
throttledBySystem: false,
|
|
};
|
|
}
|
|
|
|
export function resolveDefaultVitestPool(
|
|
_env: Record<string, string | undefined> = process.env,
|
|
): OpenClawVitestPool {
|
|
return "threads";
|
|
}
|
|
|
|
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 defaultPool = resolveDefaultVitestPool();
|
|
const localScheduling = resolveLocalVitestScheduling(
|
|
process.env,
|
|
detectVitestHostInfo(),
|
|
defaultPool,
|
|
);
|
|
const ciWorkers = isWindows ? 2 : 3;
|
|
|
|
if (!isCI && localScheduling.throttledBySystem && shouldPrintVitestThrottle(process.env)) {
|
|
console.error(
|
|
`[vitest] throttling local workers to ${localScheduling.maxWorkers}${
|
|
localScheduling.fileParallelism ? "" : " with file parallelism disabled"
|
|
} because the host already looks busy.`,
|
|
);
|
|
}
|
|
|
|
export const sharedVitestConfig = {
|
|
resolve: {
|
|
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,
|
|
unstubEnvs: true,
|
|
unstubGlobals: true,
|
|
isolate: false,
|
|
pool: defaultPool,
|
|
runner: "./test/non-isolated-runner.ts",
|
|
maxWorkers: isCI ? ciWorkers : localScheduling.maxWorkers,
|
|
fileParallelism: isCI ? true : localScheduling.fileParallelism,
|
|
forceRerunTriggers: [
|
|
"package.json",
|
|
"pnpm-lock.yaml",
|
|
"test/setup.ts",
|
|
"test/setup.shared.ts",
|
|
"test/setup.extensions.ts",
|
|
"test/setup-openclaw-runtime.ts",
|
|
"vitest.channel-paths.mjs",
|
|
"vitest.channels.config.ts",
|
|
"vitest.acp.config.ts",
|
|
"vitest.boundary.config.ts",
|
|
"vitest.bundled.config.ts",
|
|
"vitest.cli.config.ts",
|
|
"vitest.config.ts",
|
|
"vitest.contracts.config.ts",
|
|
"vitest.cron.config.ts",
|
|
"vitest.daemon.config.ts",
|
|
"vitest.e2e.config.ts",
|
|
"vitest.extension-acpx-paths.mjs",
|
|
"vitest.extension-acpx.config.ts",
|
|
"vitest.extension-bluebubbles-paths.mjs",
|
|
"vitest.extension-bluebubbles.config.ts",
|
|
"vitest.extension-channels.config.ts",
|
|
"vitest.extension-diffs-paths.mjs",
|
|
"vitest.extension-diffs.config.ts",
|
|
"vitest.extension-feishu-paths.mjs",
|
|
"vitest.extension-feishu.config.ts",
|
|
"vitest.extension-irc-paths.mjs",
|
|
"vitest.extension-irc.config.ts",
|
|
"vitest.extension-mattermost-paths.mjs",
|
|
"vitest.extension-mattermost.config.ts",
|
|
"vitest.extension-matrix-paths.mjs",
|
|
"vitest.extension-matrix.config.ts",
|
|
"vitest.extension-memory-paths.mjs",
|
|
"vitest.extension-memory.config.ts",
|
|
"vitest.extension-messaging-paths.mjs",
|
|
"vitest.extension-messaging.config.ts",
|
|
"vitest.extension-msteams-paths.mjs",
|
|
"vitest.extension-msteams.config.ts",
|
|
"vitest.extensions.config.ts",
|
|
"vitest.gateway.config.ts",
|
|
"vitest.hooks.config.ts",
|
|
"vitest.infra.config.ts",
|
|
"vitest.live.config.ts",
|
|
"vitest.media.config.ts",
|
|
"vitest.media-understanding.config.ts",
|
|
"vitest.performance-config.ts",
|
|
"vitest.scoped-config.ts",
|
|
"vitest.shared-core.config.ts",
|
|
"vitest.shared.config.ts",
|
|
"vitest.tooling.config.ts",
|
|
"vitest.tui.config.ts",
|
|
"vitest.ui.config.ts",
|
|
"vitest.utils.config.ts",
|
|
"vitest.unit.config.ts",
|
|
"vitest.unit-paths.mjs",
|
|
"vitest.runtime-config.config.ts",
|
|
"vitest.secrets.config.ts",
|
|
"vitest.plugin-sdk.config.ts",
|
|
"vitest.plugins.config.ts",
|
|
"vitest.extension-telegram-paths.mjs",
|
|
"vitest.extension-telegram.config.ts",
|
|
"vitest.extension-voice-call-paths.mjs",
|
|
"vitest.extension-voice-call.config.ts",
|
|
"vitest.extension-whatsapp-paths.mjs",
|
|
"vitest.extension-whatsapp.config.ts",
|
|
"vitest.extension-zalo-paths.mjs",
|
|
"vitest.extension-zalo.config.ts",
|
|
"vitest.extension-provider-paths.mjs",
|
|
"vitest.extension-providers.config.ts",
|
|
"vitest.logging.config.ts",
|
|
"vitest.process.config.ts",
|
|
"vitest.tasks.config.ts",
|
|
"vitest.wizard.config.ts",
|
|
],
|
|
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/dreams.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" as const,
|
|
reporter: ["text", "lcov"],
|
|
all: false,
|
|
thresholds: {
|
|
lines: 70,
|
|
functions: 70,
|
|
branches: 55,
|
|
statements: 70,
|
|
},
|
|
include: ["./src/**/*.ts"],
|
|
exclude: [
|
|
`${BUNDLED_PLUGIN_ROOT_DIR}/**`,
|
|
"apps/**",
|
|
"ui/**",
|
|
"test/**",
|
|
"src/**/*.test.ts",
|
|
"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/**",
|
|
"src/acp/**",
|
|
"src/agents/**",
|
|
"src/channels/**",
|
|
"src/gateway/**",
|
|
"src/line/**",
|
|
"src/media-understanding/**",
|
|
"src/node-host/**",
|
|
"src/plugins/**",
|
|
"src/providers/**",
|
|
"src/secrets/**",
|
|
"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",
|
|
"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",
|
|
"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",
|
|
"src/gateway/call.ts",
|
|
"src/process/tau-rpc.ts",
|
|
"src/process/exec.ts",
|
|
"src/tui/**",
|
|
"src/wizard/**",
|
|
"src/browser/**",
|
|
"src/channels/web/**",
|
|
"src/webchat/**",
|
|
"src/gateway/server.ts",
|
|
"src/gateway/client.ts",
|
|
"src/gateway/protocol/**",
|
|
"src/infra/tailscale.ts",
|
|
],
|
|
},
|
|
...loadVitestExperimentalConfig(),
|
|
},
|
|
};
|