mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-30 02:08:44 +00:00
* refactor: share talk event metric extraction * refactor: reuse shared coercion helpers * refactor: reuse shared primitive guards * refactor: reuse shared record guard * refactor: reuse shared primitive helpers * refactor: reuse shared string guards * refactor: reuse shared non-empty string guard * refactor: share plugin primitive coercion helpers * refactor: reuse plugin coercion helpers * refactor: reuse plugin coercion helpers in more plugins * refactor: reuse channel coercion helpers * refactor: reuse monitor coercion helpers * refactor: reuse provider coercion helpers * refactor: reuse core coercion helpers * refactor: reuse runtime coercion helpers * refactor: reuse helper coercion in codex paths * refactor: reuse helper coercion in runtime paths * refactor: reuse codex app-server coercion helpers * refactor: reuse codex record helpers * refactor: reuse migration and qa record helpers * refactor: reuse feishu and core helper guards * refactor: reuse browser and policy coercion helpers * refactor: reuse memory wiki record helper * refactor: share boolean coercion helpers * refactor: reuse finite number coercion * refactor: reuse trimmed string list helpers * refactor: reuse string list normalization * refactor: reuse remaining string list helpers * refactor: reuse string entry normalizer * refactor: share sorted string helpers * refactor: share string list normalization * test: preserve command registry browser imports * refactor: reuse trimmed list helpers * refactor: reuse string dedupe helpers * refactor: reuse local dedupe helpers * refactor: reuse more string dedupe helpers * refactor: reuse command string dedupe helpers * refactor: dedupe memory path lists with helper * refactor: expose string dedupe helpers to plugins * refactor: reuse core string dedupe helpers * refactor: reuse shared unique value helpers * refactor: reuse unique helpers in agent utilities * refactor: reuse unique helpers in config plumbing * refactor: reuse unique helpers in extensions * refactor: reuse unique helpers in core utilities * refactor: reuse unique helpers in qa plugins * refactor: reuse unique helpers in memory plugins * refactor: reuse unique helpers in channel plugins * refactor: reuse unique helpers in core tails * refactor: reuse unique helper in comfy workflow * refactor: reuse unique helpers in test utilities * refactor: expose unique value helper to plugins * refactor: reuse unique helpers for numeric lists * refactor: replace index dedupe filters * refactor: reuse string entry normalization * refactor: reuse string normalization in plugin helpers * refactor: reuse string normalization in extension helpers * refactor: reuse string normalization in channel parsers * refactor: reuse string normalization in memory search * refactor: reuse string normalization in provider parsers * refactor: reuse string normalization in qa helpers * refactor: reuse string normalization in infra parsers * refactor: reuse string normalization in messaging parsers * refactor: reuse string normalization in core parsers * refactor: reuse string normalization in extension parsers * refactor: reuse string normalization in remaining parsers * refactor: reuse string normalization in final parser spots * refactor: reuse string normalization in qa media helpers * refactor: reuse normalization in provider and media lists * refactor: reuse normalization for remaining set filters * refactor: reuse normalization in policy allowlists * refactor: reuse normalization in session and owner lists * refactor: centralize primitive string lists * refactor: reuse lowercase entry helpers * refactor: reuse sorted string helpers * refactor: reuse unique trimmed helpers * refactor: reuse string normalization helpers * refactor: reuse catalog string helpers * refactor: reuse remaining string helpers * refactor: simplify remaining list normalization * refactor: reuse codex auth order normalization * chore: refresh plugin sdk api baseline * fix: make shared string sorting deterministic * chore: refresh plugin sdk api baseline * fix: align host env security ordering
197 lines
6.8 KiB
TypeScript
197 lines
6.8 KiB
TypeScript
import {
|
|
asBoolean as readBoolean,
|
|
isRecord,
|
|
normalizeOptionalString as readString,
|
|
} from "openclaw/plugin-sdk/string-coerce-runtime";
|
|
import type { QaRuntimeParityTier, QaSeedScenarioWithSource } from "./scenario-catalog.js";
|
|
|
|
export type QaRuntimeToolBucket =
|
|
| "codex-native-workspace"
|
|
| "openclaw-dynamic-integration"
|
|
| "optional-profile-or-plugin";
|
|
|
|
export type QaRuntimeToolExpectedLayer =
|
|
| "codex-native-workspace"
|
|
| "openclaw-dynamic"
|
|
| "profile-or-plugin";
|
|
|
|
export type QaRuntimeCapabilityLayer =
|
|
| "codex-native-workspace"
|
|
| "openclaw-dynamic-direct"
|
|
| "openclaw-dynamic-searchable"
|
|
| "optional-profile-or-plugin"
|
|
| "structural-text";
|
|
|
|
export type QaCodexToolLoading = "direct" | "searchable";
|
|
|
|
export type RuntimeParityComparisonMode = "default" | "codex-native-workspace" | "outcome-only";
|
|
|
|
export type QaRuntimeToolCoverageMetadata = {
|
|
bucket: QaRuntimeToolBucket;
|
|
expectedLayer: QaRuntimeToolExpectedLayer;
|
|
capabilityLayer: QaRuntimeCapabilityLayer;
|
|
required: boolean;
|
|
tracking?: string;
|
|
reason?: string;
|
|
codexDefaultImpact?: string;
|
|
qaImpact?: string;
|
|
action?: string;
|
|
};
|
|
|
|
export const QA_RUNTIME_TOOL_BUCKETS: readonly QaRuntimeToolBucket[] = [
|
|
"codex-native-workspace",
|
|
"openclaw-dynamic-integration",
|
|
"optional-profile-or-plugin",
|
|
] as const;
|
|
|
|
export const QA_RUNTIME_TOOL_EXPECTED_LAYERS: readonly QaRuntimeToolExpectedLayer[] = [
|
|
"codex-native-workspace",
|
|
"openclaw-dynamic",
|
|
"profile-or-plugin",
|
|
] as const;
|
|
|
|
export const QA_RUNTIME_CAPABILITY_LAYERS: readonly QaRuntimeCapabilityLayer[] = [
|
|
"codex-native-workspace",
|
|
"openclaw-dynamic-direct",
|
|
"openclaw-dynamic-searchable",
|
|
"optional-profile-or-plugin",
|
|
"structural-text",
|
|
] as const;
|
|
|
|
export const QA_CODEX_TOOL_LOADING_MODES: readonly QaCodexToolLoading[] = [
|
|
"direct",
|
|
"searchable",
|
|
] as const;
|
|
|
|
const DEFAULT_LAYER_BY_BUCKET: Record<QaRuntimeToolBucket, QaRuntimeToolExpectedLayer> = {
|
|
"codex-native-workspace": "codex-native-workspace",
|
|
"openclaw-dynamic-integration": "openclaw-dynamic",
|
|
"optional-profile-or-plugin": "profile-or-plugin",
|
|
};
|
|
|
|
const DEFAULT_CAPABILITY_LAYER_BY_BUCKET: Record<QaRuntimeToolBucket, QaRuntimeCapabilityLayer> = {
|
|
"codex-native-workspace": "codex-native-workspace",
|
|
"openclaw-dynamic-integration": "openclaw-dynamic-searchable",
|
|
"optional-profile-or-plugin": "optional-profile-or-plugin",
|
|
};
|
|
|
|
function isQaRuntimeToolBucket(value: string): value is QaRuntimeToolBucket {
|
|
return QA_RUNTIME_TOOL_BUCKETS.includes(value as QaRuntimeToolBucket);
|
|
}
|
|
|
|
function isQaRuntimeToolExpectedLayer(value: string): value is QaRuntimeToolExpectedLayer {
|
|
return QA_RUNTIME_TOOL_EXPECTED_LAYERS.includes(value as QaRuntimeToolExpectedLayer);
|
|
}
|
|
|
|
function isQaRuntimeCapabilityLayer(value: string): value is QaRuntimeCapabilityLayer {
|
|
return QA_RUNTIME_CAPABILITY_LAYERS.includes(value as QaRuntimeCapabilityLayer);
|
|
}
|
|
|
|
export function readRuntimeToolCoverageConfig(
|
|
config: Record<string, unknown> | undefined,
|
|
): Record<string, unknown> | undefined {
|
|
return isRecord(config?.toolCoverage) ? config.toolCoverage : undefined;
|
|
}
|
|
|
|
function inferRuntimeToolBucket(params: {
|
|
config?: Record<string, unknown>;
|
|
runtimeParityTier?: QaRuntimeParityTier;
|
|
}): QaRuntimeToolBucket {
|
|
const toolCoverage = readRuntimeToolCoverageConfig(params.config);
|
|
const explicit = readString(toolCoverage?.bucket);
|
|
if (explicit) {
|
|
if (!isQaRuntimeToolBucket(explicit)) {
|
|
throw new Error(
|
|
`unknown runtime tool coverage bucket: ${explicit}; expected ${QA_RUNTIME_TOOL_BUCKETS.join(
|
|
", ",
|
|
)}`,
|
|
);
|
|
}
|
|
return explicit;
|
|
}
|
|
if (params.runtimeParityTier === "optional" || params.config?.expectedAvailable === false) {
|
|
return "optional-profile-or-plugin";
|
|
}
|
|
return "openclaw-dynamic-integration";
|
|
}
|
|
|
|
export function readRuntimeToolCoverageMetadata(params: {
|
|
config?: Record<string, unknown>;
|
|
runtimeParityTier?: QaRuntimeParityTier;
|
|
}): QaRuntimeToolCoverageMetadata {
|
|
const toolCoverage = readRuntimeToolCoverageConfig(params.config);
|
|
const bucket = inferRuntimeToolBucket(params);
|
|
const expectedLayerInput = readString(toolCoverage?.expectedLayer);
|
|
if (expectedLayerInput && !isQaRuntimeToolExpectedLayer(expectedLayerInput)) {
|
|
throw new Error(
|
|
`unknown runtime tool expectedLayer: ${expectedLayerInput}; expected ${QA_RUNTIME_TOOL_EXPECTED_LAYERS.join(
|
|
", ",
|
|
)}`,
|
|
);
|
|
}
|
|
const expectedLayer = expectedLayerInput
|
|
? (expectedLayerInput as QaRuntimeToolExpectedLayer)
|
|
: DEFAULT_LAYER_BY_BUCKET[bucket];
|
|
const capabilityLayerInput = readString(toolCoverage?.capabilityLayer);
|
|
if (capabilityLayerInput && !isQaRuntimeCapabilityLayer(capabilityLayerInput)) {
|
|
throw new Error(
|
|
`unknown runtime tool capabilityLayer: ${capabilityLayerInput}; expected ${QA_RUNTIME_CAPABILITY_LAYERS.join(
|
|
", ",
|
|
)}`,
|
|
);
|
|
}
|
|
const capabilityLayer = capabilityLayerInput
|
|
? (capabilityLayerInput as QaRuntimeCapabilityLayer)
|
|
: DEFAULT_CAPABILITY_LAYER_BY_BUCKET[bucket];
|
|
const explicitSearchableDynamic = capabilityLayerInput === "openclaw-dynamic-searchable";
|
|
const required =
|
|
readBoolean(toolCoverage?.required) ??
|
|
(bucket !== "optional-profile-or-plugin" && !explicitSearchableDynamic);
|
|
return {
|
|
bucket,
|
|
expectedLayer,
|
|
capabilityLayer,
|
|
required,
|
|
...((readString(toolCoverage?.tracking) ?? readString(toolCoverage?.issue))
|
|
? { tracking: readString(toolCoverage?.tracking) ?? readString(toolCoverage?.issue) }
|
|
: {}),
|
|
...(readString(toolCoverage?.reason) ? { reason: readString(toolCoverage?.reason) } : {}),
|
|
...(readString(toolCoverage?.codexDefaultImpact)
|
|
? { codexDefaultImpact: readString(toolCoverage?.codexDefaultImpact) }
|
|
: {}),
|
|
...(readString(toolCoverage?.qaImpact) ? { qaImpact: readString(toolCoverage?.qaImpact) } : {}),
|
|
...(readString(toolCoverage?.action) ? { action: readString(toolCoverage?.action) } : {}),
|
|
};
|
|
}
|
|
|
|
export function readScenarioRuntimeToolCoverageMetadata(
|
|
scenario: QaSeedScenarioWithSource,
|
|
): QaRuntimeToolCoverageMetadata {
|
|
return readRuntimeToolCoverageMetadata({
|
|
config: scenario.execution.config,
|
|
runtimeParityTier: scenario.runtimeParityTier,
|
|
});
|
|
}
|
|
|
|
export function runtimeToolComparisonModeForScenario(
|
|
scenario: QaSeedScenarioWithSource,
|
|
): RuntimeParityComparisonMode {
|
|
const explicit = readString(scenario.execution.config?.runtimeParityComparison);
|
|
if (explicit) {
|
|
if (
|
|
explicit !== "default" &&
|
|
explicit !== "codex-native-workspace" &&
|
|
explicit !== "outcome-only"
|
|
) {
|
|
throw new Error(
|
|
`unknown runtime parity comparison mode: ${explicit}; expected default, codex-native-workspace, outcome-only`,
|
|
);
|
|
}
|
|
return explicit;
|
|
}
|
|
return readScenarioRuntimeToolCoverageMetadata(scenario).expectedLayer ===
|
|
"codex-native-workspace"
|
|
? "codex-native-workspace"
|
|
: "default";
|
|
}
|