mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-28 10:02:15 +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
150 lines
4.6 KiB
TypeScript
150 lines
4.6 KiB
TypeScript
import { normalizeStringEntries, uniqueStrings } from "openclaw/plugin-sdk/string-coerce-runtime";
|
|
import {
|
|
definePluginEntry,
|
|
type ProviderAuthContext,
|
|
type ProviderAuthResult,
|
|
} from "./runtime-api.js";
|
|
|
|
const DEFAULT_BASE_URL = "http://localhost:3000/v1";
|
|
const DEFAULT_API_KEY = "n/a";
|
|
const DEFAULT_CONTEXT_WINDOW = 128_000;
|
|
const DEFAULT_MAX_TOKENS = 8192;
|
|
const DEFAULT_MODEL_IDS = [
|
|
"gpt-5.2",
|
|
"gpt-5.2-codex",
|
|
"gpt-5.1",
|
|
"gpt-5.1-codex",
|
|
"gpt-5.1-codex-max",
|
|
"gpt-5-mini",
|
|
"claude-opus-4.6",
|
|
"claude-opus-4.7",
|
|
"claude-sonnet-4.6",
|
|
"gemini-3-pro",
|
|
"gemini-3-flash",
|
|
] as const;
|
|
|
|
function normalizeBaseUrl(value: string): string {
|
|
const trimmed = value.trim();
|
|
if (!trimmed) {
|
|
return DEFAULT_BASE_URL;
|
|
}
|
|
let normalized = trimmed;
|
|
while (normalized.endsWith("/")) {
|
|
normalized = normalized.slice(0, -1);
|
|
}
|
|
if (!normalized.endsWith("/v1")) {
|
|
normalized = `${normalized}/v1`;
|
|
}
|
|
return normalized;
|
|
}
|
|
|
|
function validateBaseUrl(value: string): string | undefined {
|
|
const normalized = normalizeBaseUrl(value);
|
|
return URL.canParse(normalized) ? undefined : "Enter a valid URL";
|
|
}
|
|
|
|
function parseModelIds(input: string): string[] {
|
|
const parsed = normalizeStringEntries(input.split(/[\n,]/));
|
|
return uniqueStrings(parsed);
|
|
}
|
|
|
|
function buildModelDefinition(modelId: string) {
|
|
return {
|
|
id: modelId,
|
|
name: modelId,
|
|
api: "openai-completions" as const,
|
|
reasoning: false,
|
|
input: ["text", "image"] as Array<"text" | "image">,
|
|
cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
|
|
contextWindow: DEFAULT_CONTEXT_WINDOW,
|
|
maxTokens: DEFAULT_MAX_TOKENS,
|
|
};
|
|
}
|
|
|
|
export default definePluginEntry({
|
|
id: "copilot-proxy",
|
|
name: "Copilot Proxy",
|
|
description: "Local Copilot Proxy (VS Code LM) provider plugin",
|
|
register(api) {
|
|
api.registerProvider({
|
|
id: "copilot-proxy",
|
|
label: "Copilot Proxy",
|
|
docsPath: "/providers/models",
|
|
auth: [
|
|
{
|
|
id: "local",
|
|
label: "Local proxy",
|
|
hint: "Configure base URL + models for the Copilot Proxy server",
|
|
kind: "custom",
|
|
run: async (ctx: ProviderAuthContext): Promise<ProviderAuthResult> => {
|
|
const baseUrlInput = await ctx.prompter.text({
|
|
message: "Copilot Proxy base URL",
|
|
initialValue: DEFAULT_BASE_URL,
|
|
validate: validateBaseUrl,
|
|
});
|
|
|
|
const modelInput = await ctx.prompter.text({
|
|
message: "Model IDs (comma-separated)",
|
|
initialValue: DEFAULT_MODEL_IDS.join(", "),
|
|
validate: (value: string) =>
|
|
parseModelIds(value).length > 0 ? undefined : "Enter at least one model id",
|
|
});
|
|
|
|
const baseUrl = normalizeBaseUrl(baseUrlInput);
|
|
const modelIds = parseModelIds(modelInput);
|
|
const defaultModelId = modelIds[0] ?? DEFAULT_MODEL_IDS[0];
|
|
const defaultModelRef = `copilot-proxy/${defaultModelId}`;
|
|
|
|
return {
|
|
profiles: [
|
|
{
|
|
profileId: "copilot-proxy:local",
|
|
credential: {
|
|
type: "token",
|
|
provider: "copilot-proxy",
|
|
token: DEFAULT_API_KEY,
|
|
},
|
|
},
|
|
],
|
|
configPatch: {
|
|
models: {
|
|
providers: {
|
|
"copilot-proxy": {
|
|
baseUrl,
|
|
apiKey: DEFAULT_API_KEY,
|
|
api: "openai-completions",
|
|
authHeader: false,
|
|
models: modelIds.map((modelId) => buildModelDefinition(modelId)),
|
|
},
|
|
},
|
|
},
|
|
agents: {
|
|
defaults: {
|
|
models: Object.fromEntries(
|
|
modelIds.map((modelId) => [`copilot-proxy/${modelId}`, {}]),
|
|
),
|
|
},
|
|
},
|
|
},
|
|
defaultModel: defaultModelRef,
|
|
notes: [
|
|
"Start the Copilot Proxy VS Code extension before using these models.",
|
|
"Copilot Proxy serves /v1/chat/completions; base URL must include /v1.",
|
|
"Model availability depends on your Copilot plan; edit models.providers.copilot-proxy if needed.",
|
|
],
|
|
};
|
|
},
|
|
},
|
|
],
|
|
wizard: {
|
|
setup: {
|
|
choiceId: "copilot-proxy",
|
|
choiceLabel: "Copilot Proxy",
|
|
choiceHint: "Configure base URL + model ids",
|
|
methodId: "local",
|
|
},
|
|
},
|
|
});
|
|
},
|
|
});
|