mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-28 19:15:52 +00:00
* refactor: extract agent core package Introduce packages/agent-core as the OpenClaw-owned home for reusable agent loop, harness, session, prompt, and runtime dependency contracts. * refactor: extract shared llm runtime Move provider model registries, stream wrappers, OAuth helpers, and LLM utilities into src/llm with plugin-sdk barrels instead of depending on the old embedded runtime layout. * refactor: remove pi runtime internals Rename remaining Pi-shaped agent surfaces to OpenClaw agent runtime names, delete obsolete Pi docs and package graph checks, and add the third-party notice for incorporated code. * refactor: tighten agent session runtime Make agent-core/runtime dependencies explicit, consolidate compaction and session transcript helpers, and move model/session helpers behind OpenClaw-owned contracts. * refactor: remove static model and pi auth paths Drop static model catalogs and Pi auth bridges, move model/provider facts to manifest-owned runtime contracts, and harden internal embedded-agent utilities. * refactor: remove legacy provider compat paths * docs: remove agent parity notes * fix: skip provider wildcard metadata parsing * refactor: share session extension sdk loading * refactor: inline acpx proxy error formatter * refactor: fold edit recovery into edit tool * fix: accept extension batch separator * test: align startup provider plugin expectations * fix: restore provider-scoped release discovery * test: align static asset packaging expectations * fix: run static provider catalogs during scoped discovery * fix: add provider entry catalogs for scoped live discovery * fix: load lightweight provider catalog entries * fix: refresh provider-scoped plugin metadata * fix: keep provider catalog entries on release live path * fix: keep static manifest models in release live checks * fix: harden release model discovery * fix: reduce OpenAI live cache probe reasoning * fix: disable OpenAI cache probe reasoning * ci: extend OpenAI gateway live timeout * fix: extend live gateway model budget * fix: stabilize release validation regressions * fix: honor provider aliases in model rows * fix: stabilize release validation lanes * fix: stabilize release memory qa * ci: stabilize release validation lanes * ci: prefer ipv4 for live docker node calls * fix: restore shared tool-call stream wrapper * ci: remove legacy pi test shard alias * fix: clean up embedded agent test drift * fix: stabilize runtime alias status * fix: clean up embedded agent ci drift * fix: restore release ci invariants * fix: clean up post-rebase runtime drift * fix: restore release ci checks * fix: restore release ci after rebase * fix: remove stale pi runtime path * test: align compaction runtime expectations * test: update plugin prerelease expectations * fix: handle claude live tool approvals * fix: stabilize release validation gates * fix: finish agent runtime import * test: finish post-rebase agent runtime mocks * fix: keep codex compaction native * fix: stabilize codex app-server hook tests * test: isolate codex diagnostic active run * test: remove codex diagnostic completion race # Conflicts: # extensions/codex/src/app-server/run-attempt.test.ts * ci: fix full release manifest performance run id * refactor: narrow llm plugin sdk boundary * chore: drop generated google boundary stamps * fix: repair rebase fallout * fix: clean up rebased runtime references * fix: decode codex jwt payloads as base64url * fix: preserve shipped pi runtime alias * fix: add scoped sdk virtual modules * fix: decode llm codex oauth jwt as base64url * fix: avoid stale vertex adc negative cache * fix: harden tool arg decoding and codeql path * fix: keep vertex adc negative checks live * refactor: consolidate codex jwt and edit helpers * fix: await codex oauth node runtime imports * fix: preserve sdk tool and notice contracts * fix: preserve shipped compat config boundaries * fix: align codex oauth callback host * fix: terminate agent-core loop streams on failure * fix: keep codex oauth callback alive during fallback * ci: include session tools in critical codeql scans * fix: keep Cloudflare Anthropic provider auth header * docs: redirect legacy pi runtime pages * fix: honor bundled web provider compat discovery * fix: protect session output spill files * fix: keep legacy agent dir env blocked * fix: contain auto-discovered skill symlinks * fix: harden agent core sdk proxy surfaces * fix: restore approval reaction sdk compat * fix: keep live docker runs bounded * fix: keep codex oauth redirect host aligned * fix: resolve post-rebase agent runtime drift * fix: redact anthropic oauth parse failures * fix: preserve responses strict tool shaping * fix: repair agent runtime rebase cleanup * docs: redirect retired parity pages * fix: bound auto-discovered resources to roots * fix: repair post-rebase agent test drift * fix: preserve bundled provider allowlist migration * fix: preserve manifest-owned provider aliases * fix: declare photon image dependency * fix: keep provider headers out of proxy body * fix: preserve shipped env aliases * fix: refresh control ui i18n generated state * fix: quote read fallback paths * fix: preview edits through configured backend * test: satisfy core test typecheck * fix: preserve ZAI usage auth fallback * test: repair codex diagnostic test * fix: repair agent runtime rebase drift * test: finish embedded runner import rename * fix: repair agent runtime rebase integrations * test: align compaction oauth fallback expectations * fix: allow sdk-auth session models * fix: update doctor tool schema import * fix: preserve bedrock plugin region * fix: stream harmony-like prose immediately * ci: include session runtime in codeql shards * fix: repair latest rebase integrations * fix: honor explicit codex websocket transport * fix: keep openai-compatible credentials provider-scoped * fix: refresh sdk api baseline after rebase * fix: route cli runtime aliases through openclaw harness * test: rename stale harness mock expectation * test: rename embedded agent overflow calls * test: clean embedded auth test wording * test: use openclaw stream types in deepinfra cache test * fix: refresh sdk api baseline on latest main * fix: honor bundled discovery compat allowlists * fix: refresh sdk api baseline after latest rebase * fix: remove stale rebase imports * test: rename stale model catalog mock * test: mock renamed doctor runtime modules * fix: map canonical kimi env auth * fix: use internal model registry in bench script * fix: migrate deepinfra provider catalog entry * fix: enforce builtin tool suppression * fix: route compaction auth and proxy payloads safely * refactor: prune unused llm registry leftovers * test: update codex hooks session import * test: fix model picker ci coverage * test: align model picker auth mock types
492 lines
15 KiB
TypeScript
492 lines
15 KiB
TypeScript
import type { OpenClawConfig } from "openclaw/plugin-sdk/config-contracts";
|
|
import { resolvePluginConfigObject } from "openclaw/plugin-sdk/plugin-config-runtime";
|
|
import {
|
|
definePluginEntry,
|
|
type ProviderCatalogContext,
|
|
type ProviderCatalogResult,
|
|
type ProviderAuthContext,
|
|
type ProviderAuthResult,
|
|
type ProviderAuthMethodNonInteractiveContext,
|
|
type UnifiedModelCatalogEntry,
|
|
type UnifiedModelCatalogProviderContext,
|
|
} from "openclaw/plugin-sdk/plugin-entry";
|
|
import {
|
|
applyAuthProfileConfig,
|
|
coerceSecretRef,
|
|
ensureAuthProfileStore,
|
|
listProfilesForProvider,
|
|
normalizeOptionalSecretInput,
|
|
resolveDefaultSecretProviderAlias,
|
|
upsertAuthProfileWithLock,
|
|
} from "openclaw/plugin-sdk/provider-auth";
|
|
import { getCachedLiveCatalogValue } from "openclaw/plugin-sdk/provider-catalog-shared";
|
|
import { normalizeOptionalLowercaseString } from "openclaw/plugin-sdk/string-coerce-runtime";
|
|
import { resolveFirstGithubToken } from "./auth.js";
|
|
import { githubCopilotMemoryEmbeddingProviderAdapter } from "./embeddings.js";
|
|
import {
|
|
PROVIDER_ID,
|
|
fetchCopilotModelCatalog,
|
|
resolveCopilotForwardCompatModel,
|
|
} from "./models.js";
|
|
import { buildGithubCopilotReplayPolicy } from "./replay-policy.js";
|
|
import { wrapCopilotProviderStream } from "./stream.js";
|
|
|
|
const COPILOT_ENV_VARS = ["COPILOT_GITHUB_TOKEN", "GH_TOKEN", "GITHUB_TOKEN"];
|
|
const DEFAULT_COPILOT_MODEL = "github-copilot/claude-opus-4.7";
|
|
const DEFAULT_COPILOT_PROFILE_ID = "github-copilot:github";
|
|
const COPILOT_XHIGH_MODEL_IDS = ["gpt-5.4", "gpt-5.3-codex"] as const;
|
|
|
|
type GithubCopilotPluginConfig = {
|
|
discovery?: {
|
|
enabled?: boolean;
|
|
};
|
|
};
|
|
|
|
async function loadGithubCopilotRuntime() {
|
|
return await import("./register.runtime.js");
|
|
}
|
|
|
|
function applyCopilotDefaultModel(cfg: OpenClawConfig): OpenClawConfig {
|
|
const defaults = cfg.agents?.defaults;
|
|
const existingModel = defaults?.model;
|
|
const existingPrimary =
|
|
typeof existingModel === "string"
|
|
? existingModel.trim()
|
|
: typeof existingModel === "object" && typeof existingModel?.primary === "string"
|
|
? existingModel.primary.trim()
|
|
: "";
|
|
if (existingPrimary) {
|
|
return cfg;
|
|
}
|
|
const fallbacks =
|
|
typeof existingModel === "object" && existingModel !== null && "fallbacks" in existingModel
|
|
? (existingModel as { fallbacks?: string[] }).fallbacks
|
|
: undefined;
|
|
return {
|
|
...cfg,
|
|
agents: {
|
|
...cfg.agents,
|
|
defaults: {
|
|
...defaults,
|
|
model: {
|
|
...(fallbacks ? { fallbacks } : undefined),
|
|
primary: DEFAULT_COPILOT_MODEL,
|
|
},
|
|
models: {
|
|
...defaults?.models,
|
|
[DEFAULT_COPILOT_MODEL]: defaults?.models?.[DEFAULT_COPILOT_MODEL] ?? {},
|
|
},
|
|
},
|
|
},
|
|
};
|
|
}
|
|
|
|
function resolveExistingCopilotTokenProfileId(agentDir?: string): string | undefined {
|
|
const authStore = ensureAuthProfileStore(agentDir, {
|
|
allowKeychainPrompt: false,
|
|
});
|
|
return listProfilesForProvider(authStore, PROVIDER_ID).find((profileId) => {
|
|
const profile = authStore.profiles[profileId];
|
|
if (profile?.type !== "token") {
|
|
return false;
|
|
}
|
|
return Boolean(
|
|
normalizeOptionalSecretInput(profile.token) || coerceSecretRef(profile.tokenRef)?.id.trim(),
|
|
);
|
|
});
|
|
}
|
|
|
|
function resolveExistingCopilotAuthResult(agentDir?: string): ProviderAuthResult | null {
|
|
const profileId = resolveExistingCopilotTokenProfileId(agentDir);
|
|
if (!profileId) {
|
|
return null;
|
|
}
|
|
const authStore = ensureAuthProfileStore(agentDir, {
|
|
allowKeychainPrompt: false,
|
|
});
|
|
const credential = authStore.profiles[profileId];
|
|
if (!credential || credential.type !== "token") {
|
|
return null;
|
|
}
|
|
return {
|
|
profiles: [
|
|
{
|
|
profileId,
|
|
credential,
|
|
},
|
|
],
|
|
defaultModel: DEFAULT_COPILOT_MODEL,
|
|
};
|
|
}
|
|
|
|
async function resolveCopilotNonInteractiveToken(
|
|
ctx: ProviderAuthMethodNonInteractiveContext,
|
|
flagValue: string | undefined,
|
|
) {
|
|
const resolveFromEnvChain = async () => {
|
|
for (const envVar of COPILOT_ENV_VARS) {
|
|
const resolved = await ctx.resolveApiKey({
|
|
provider: PROVIDER_ID,
|
|
flagName: "--github-copilot-token",
|
|
envVar,
|
|
envVarName: envVar,
|
|
allowProfile: false,
|
|
required: false,
|
|
});
|
|
if (resolved) {
|
|
return resolved;
|
|
}
|
|
}
|
|
return null;
|
|
};
|
|
|
|
if (ctx.opts.secretInputMode === "ref") {
|
|
const resolved = await resolveFromEnvChain();
|
|
if (resolved) {
|
|
return resolved;
|
|
}
|
|
if (flagValue) {
|
|
ctx.runtime.error(
|
|
[
|
|
"--github-copilot-token cannot be used with --secret-input-mode ref unless COPILOT_GITHUB_TOKEN, GH_TOKEN, or GITHUB_TOKEN is set in env.",
|
|
"Set one of those env vars and omit --github-copilot-token, or use --secret-input-mode plaintext.",
|
|
].join("\n"),
|
|
);
|
|
ctx.runtime.exit(1);
|
|
}
|
|
return null;
|
|
}
|
|
|
|
const primary = await ctx.resolveApiKey({
|
|
provider: PROVIDER_ID,
|
|
flagValue,
|
|
flagName: "--github-copilot-token",
|
|
envVar: COPILOT_ENV_VARS[0],
|
|
envVarName: COPILOT_ENV_VARS[0],
|
|
allowProfile: false,
|
|
required: false,
|
|
});
|
|
if (primary || flagValue) {
|
|
return primary;
|
|
}
|
|
|
|
for (const envVar of COPILOT_ENV_VARS.slice(1)) {
|
|
const resolved = await ctx.resolveApiKey({
|
|
provider: PROVIDER_ID,
|
|
flagName: "--github-copilot-token",
|
|
envVar,
|
|
envVarName: envVar,
|
|
allowProfile: false,
|
|
required: false,
|
|
});
|
|
if (resolved) {
|
|
return resolved;
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
async function runGitHubCopilotNonInteractiveAuth(
|
|
ctx: ProviderAuthMethodNonInteractiveContext,
|
|
): Promise<OpenClawConfig | null> {
|
|
const opts = ctx.opts as Record<string, unknown> | undefined;
|
|
const flagValue = normalizeOptionalSecretInput(opts?.githubCopilotToken);
|
|
const resolved = await resolveCopilotNonInteractiveToken(ctx, flagValue);
|
|
|
|
let profileId = DEFAULT_COPILOT_PROFILE_ID;
|
|
if (resolved) {
|
|
const useTokenRef = ctx.opts.secretInputMode === "ref" && resolved.source === "env";
|
|
if (useTokenRef && !resolved.envVarName) {
|
|
ctx.runtime.error(
|
|
[
|
|
'--secret-input-mode ref requires an explicit environment variable for provider "github-copilot".',
|
|
"Set COPILOT_GITHUB_TOKEN in env and retry, or use --secret-input-mode plaintext.",
|
|
].join("\n"),
|
|
);
|
|
ctx.runtime.exit(1);
|
|
return null;
|
|
}
|
|
await upsertAuthProfileWithLock({
|
|
profileId,
|
|
credential: {
|
|
type: "token",
|
|
provider: PROVIDER_ID,
|
|
...(useTokenRef
|
|
? {
|
|
tokenRef: {
|
|
source: "env",
|
|
provider: resolveDefaultSecretProviderAlias(ctx.baseConfig, "env", {
|
|
preferFirstProviderForSource: true,
|
|
}),
|
|
id: resolved.envVarName!,
|
|
},
|
|
}
|
|
: { token: resolved.key }),
|
|
},
|
|
agentDir: ctx.agentDir,
|
|
});
|
|
} else {
|
|
if (flagValue && ctx.opts.secretInputMode === "ref") {
|
|
return null;
|
|
}
|
|
const existingProfileId = resolveExistingCopilotTokenProfileId(ctx.agentDir);
|
|
if (!existingProfileId) {
|
|
ctx.runtime.error(
|
|
"Missing --github-copilot-token (or COPILOT_GITHUB_TOKEN / GH_TOKEN / GITHUB_TOKEN env var) for --auth-choice github-copilot.",
|
|
);
|
|
ctx.runtime.exit(1);
|
|
return null;
|
|
}
|
|
profileId = existingProfileId;
|
|
}
|
|
|
|
return applyCopilotDefaultModel(
|
|
applyAuthProfileConfig(ctx.config, {
|
|
profileId,
|
|
provider: PROVIDER_ID,
|
|
mode: "token",
|
|
}),
|
|
);
|
|
}
|
|
|
|
export default definePluginEntry({
|
|
id: "github-copilot",
|
|
name: "GitHub Copilot Provider",
|
|
description: "Bundled GitHub Copilot provider plugin",
|
|
register(api) {
|
|
const startupPluginConfig = (api.pluginConfig ?? {}) as GithubCopilotPluginConfig;
|
|
|
|
function resolveCurrentPluginConfig(config?: OpenClawConfig): GithubCopilotPluginConfig {
|
|
const runtimePluginConfig = resolvePluginConfigObject(config, "github-copilot");
|
|
if (runtimePluginConfig) {
|
|
return runtimePluginConfig as GithubCopilotPluginConfig;
|
|
}
|
|
return config ? {} : startupPluginConfig;
|
|
}
|
|
|
|
async function runGithubCopilotCatalog(
|
|
ctx: ProviderCatalogContext,
|
|
): Promise<ProviderCatalogResult> {
|
|
const pluginConfig = resolveCurrentPluginConfig(ctx.config);
|
|
const discoveryEnabled = pluginConfig.discovery?.enabled;
|
|
if (discoveryEnabled === false) {
|
|
return null;
|
|
}
|
|
const { DEFAULT_COPILOT_API_BASE_URL, resolveCopilotApiToken } =
|
|
await loadGithubCopilotRuntime();
|
|
const { githubToken, hasProfile } = await resolveFirstGithubToken({
|
|
agentDir: ctx.agentDir,
|
|
config: ctx.config,
|
|
env: ctx.env,
|
|
});
|
|
if (!hasProfile && !githubToken) {
|
|
return null;
|
|
}
|
|
let baseUrl = DEFAULT_COPILOT_API_BASE_URL;
|
|
let copilotApiToken: string | undefined;
|
|
if (githubToken) {
|
|
try {
|
|
const token = await resolveCopilotApiToken({
|
|
githubToken,
|
|
env: ctx.env,
|
|
});
|
|
baseUrl = token.baseUrl;
|
|
copilotApiToken = token.token;
|
|
} catch {
|
|
baseUrl = DEFAULT_COPILOT_API_BASE_URL;
|
|
}
|
|
}
|
|
// Try to fetch the live model catalog from Copilot's /models endpoint so
|
|
// the runtime tracks per-account entitlements and accurate context
|
|
// windows (max_context_window_tokens) without manifest churn. On any
|
|
// failure we return an empty model list, which lets the static manifest
|
|
// catalog continue to be the visible fallback for users.
|
|
let discoveredModels: Awaited<ReturnType<typeof fetchCopilotModelCatalog>> = [];
|
|
if (copilotApiToken) {
|
|
try {
|
|
discoveredModels = await getCachedLiveCatalogValue({
|
|
keyParts: [PROVIDER_ID, "models", baseUrl, copilotApiToken],
|
|
load: async () =>
|
|
await fetchCopilotModelCatalog({
|
|
copilotApiToken,
|
|
baseUrl,
|
|
}),
|
|
});
|
|
} catch {
|
|
discoveredModels = [];
|
|
}
|
|
}
|
|
return {
|
|
provider: {
|
|
baseUrl,
|
|
models: discoveredModels,
|
|
},
|
|
};
|
|
}
|
|
|
|
async function runGithubCopilotUnifiedLiveCatalog(
|
|
ctx: UnifiedModelCatalogProviderContext,
|
|
): Promise<UnifiedModelCatalogEntry[] | null> {
|
|
const result = await runGithubCopilotCatalog(ctx);
|
|
if (!result || !("provider" in result)) {
|
|
return null;
|
|
}
|
|
return (result.provider.models ?? []).map((model) => {
|
|
const entry: UnifiedModelCatalogEntry = {
|
|
kind: "text",
|
|
provider: PROVIDER_ID,
|
|
model: model.id,
|
|
source: "live",
|
|
};
|
|
if (model.name) {
|
|
entry.label = model.name;
|
|
}
|
|
return entry;
|
|
});
|
|
}
|
|
|
|
async function runGitHubCopilotAuth(ctx: ProviderAuthContext) {
|
|
const existing = resolveExistingCopilotAuthResult(ctx.agentDir);
|
|
if (existing) {
|
|
const runLogin = await ctx.prompter.confirm({
|
|
message: "GitHub Copilot auth already exists. Re-run login?",
|
|
initialValue: false,
|
|
});
|
|
if (!runLogin) {
|
|
return existing;
|
|
}
|
|
}
|
|
|
|
await ctx.prompter.note(
|
|
[
|
|
"This will open a GitHub device login to authorize Copilot.",
|
|
"Requires an active GitHub Copilot subscription.",
|
|
].join("\n"),
|
|
"GitHub Copilot",
|
|
);
|
|
|
|
const { runGitHubCopilotDeviceFlow } = await import("./login.js");
|
|
|
|
const result = await runGitHubCopilotDeviceFlow({
|
|
showCode: async ({ verificationUrl, userCode, expiresInMs }) => {
|
|
const expiresInMinutes = Math.max(1, Math.round(expiresInMs / 60_000));
|
|
await ctx.prompter.note(
|
|
[
|
|
"Open this URL in your browser and enter the code below.",
|
|
`URL: ${verificationUrl}`,
|
|
`Code: ${userCode}`,
|
|
`Code expires in ${expiresInMinutes} minutes. Never share it.`,
|
|
"",
|
|
"If a browser does not open automatically after you continue, copy the URL manually.",
|
|
].join("\n"),
|
|
"Authorize GitHub Copilot",
|
|
);
|
|
},
|
|
openUrl: async (url) => {
|
|
await ctx.openUrl(url);
|
|
},
|
|
});
|
|
|
|
if (result.status === "access_denied") {
|
|
await ctx.prompter.note("GitHub Copilot login was cancelled.", "GitHub Copilot");
|
|
return { profiles: [] };
|
|
}
|
|
|
|
if (result.status === "expired") {
|
|
await ctx.prompter.note(
|
|
"The GitHub device code expired. Retry login to get a new code.",
|
|
"GitHub Copilot",
|
|
);
|
|
return { profiles: [] };
|
|
}
|
|
|
|
return {
|
|
profiles: [
|
|
{
|
|
profileId: DEFAULT_COPILOT_PROFILE_ID,
|
|
credential: {
|
|
type: "token" as const,
|
|
provider: PROVIDER_ID,
|
|
token: result.accessToken,
|
|
},
|
|
},
|
|
],
|
|
defaultModel: DEFAULT_COPILOT_MODEL,
|
|
};
|
|
}
|
|
|
|
api.registerMemoryEmbeddingProvider(githubCopilotMemoryEmbeddingProviderAdapter);
|
|
|
|
api.registerProvider({
|
|
id: PROVIDER_ID,
|
|
label: "GitHub Copilot",
|
|
docsPath: "/providers/models",
|
|
envVars: COPILOT_ENV_VARS,
|
|
auth: [
|
|
{
|
|
id: "device",
|
|
label: "GitHub device login",
|
|
hint: "Browser device-code flow",
|
|
kind: "device_code",
|
|
run: async (ctx) => await runGitHubCopilotAuth(ctx),
|
|
runNonInteractive: async (ctx) => await runGitHubCopilotNonInteractiveAuth(ctx),
|
|
},
|
|
],
|
|
wizard: {
|
|
setup: {
|
|
choiceId: "github-copilot",
|
|
choiceLabel: "GitHub Copilot",
|
|
choiceHint: "Device login with your GitHub account",
|
|
methodId: "device",
|
|
modelSelection: {
|
|
promptWhenAuthChoiceProvided: true,
|
|
},
|
|
},
|
|
},
|
|
catalog: {
|
|
order: "late",
|
|
run: runGithubCopilotCatalog,
|
|
},
|
|
resolveDynamicModel: (ctx) => resolveCopilotForwardCompatModel(ctx),
|
|
wrapStreamFn: wrapCopilotProviderStream,
|
|
buildReplayPolicy: ({ modelId }) => buildGithubCopilotReplayPolicy(modelId),
|
|
resolveThinkingProfile: ({ modelId }) => ({
|
|
levels: [
|
|
{ id: "off" },
|
|
{ id: "minimal" },
|
|
{ id: "low" },
|
|
{ id: "medium" },
|
|
{ id: "high" },
|
|
...(COPILOT_XHIGH_MODEL_IDS.includes(
|
|
(normalizeOptionalLowercaseString(modelId) ?? "") as never,
|
|
)
|
|
? [{ id: "xhigh" as const }]
|
|
: []),
|
|
],
|
|
}),
|
|
prepareRuntimeAuth: async (ctx) => {
|
|
const { resolveCopilotApiToken } = await loadGithubCopilotRuntime();
|
|
const token = await resolveCopilotApiToken({
|
|
githubToken: ctx.apiKey,
|
|
env: ctx.env,
|
|
});
|
|
return {
|
|
apiKey: token.token,
|
|
baseUrl: token.baseUrl,
|
|
expiresAt: token.expiresAt,
|
|
};
|
|
},
|
|
resolveUsageAuth: async (ctx) => await ctx.resolveOAuthToken(),
|
|
fetchUsageSnapshot: async (ctx) => {
|
|
const { fetchCopilotUsage } = await loadGithubCopilotRuntime();
|
|
return await fetchCopilotUsage(ctx.token, ctx.timeoutMs, ctx.fetchFn);
|
|
},
|
|
});
|
|
api.registerModelCatalogProvider({
|
|
provider: PROVIDER_ID,
|
|
kinds: ["text"],
|
|
liveCatalog: runGithubCopilotUnifiedLiveCatalog,
|
|
});
|
|
},
|
|
});
|