OmniRoute/tests/unit/executor-github.test.ts
backryun a16e47c593
fix(providers): refresh web client user agents (#1699)
Integrated release/v3.7.2 changes — refreshed web client user agents, env docs, Gemini OAuth fix
2026-04-28 02:41:19 -03:00

285 lines
8.9 KiB
TypeScript

import test from "node:test";
import assert from "node:assert/strict";
import { GithubExecutor } from "../../open-sse/executors/github.ts";
import { PROVIDER_MODELS } from "../../open-sse/config/providerModels.ts";
function registerModel(provider, model) {
PROVIDER_MODELS[provider] = [...(PROVIDER_MODELS[provider] || []), model];
}
test("GithubExecutor.buildUrl routes response-format models to /responses", () => {
const originalModels = [...(PROVIDER_MODELS.gh || [])];
registerModel("gh", {
id: "gpt-4.1-responses",
name: "GPT 4.1 Responses",
targetFormat: "openai-responses",
});
try {
const executor = new GithubExecutor();
const url = executor.buildUrl("gpt-4.1-responses", true);
assert.equal(url, "https://api.githubcopilot.com/responses");
} finally {
PROVIDER_MODELS.gh = originalModels;
}
});
test("GithubExecutor.transformRequest injects JSON response instructions for Claude and strips reasoning fields", () => {
const executor = new GithubExecutor();
const body = {
response_format: {
type: "json_object",
},
messages: [
{ role: "user", content: "Return JSON" },
{
role: "assistant",
content: "draft",
reasoning_text: "internal",
reasoning_content: "internal",
},
],
};
const result = executor.transformRequest("claude-sonnet-4", body, true, {});
assert.equal(result.response_format, undefined);
assert.equal(result.messages[0].role, "system");
assert.match(result.messages[0].content, /Respond only with valid JSON/);
assert.equal(result.messages[2].reasoning_text, undefined);
assert.equal(result.messages[2].reasoning_content, undefined);
});
test("GithubExecutor.buildHeaders prefers Copilot token and sets GitHub-specific headers", () => {
const executor = new GithubExecutor();
const headers = executor.buildHeaders(
{
accessToken: "gh-access-token",
providerSpecificData: { copilotToken: "copilot-token" },
},
true
);
assert.equal(headers.Authorization, "Bearer copilot-token");
assert.equal(headers.Accept, "text/event-stream");
assert.equal(headers["editor-version"], "vscode/1.117.0");
assert.equal(headers["editor-plugin-version"], "copilot-chat/0.45.1");
assert.equal(headers["user-agent"], "GitHubCopilotChat/0.45.1");
assert.equal(headers["x-github-api-version"], "2025-04-01");
assert.equal(headers["openai-intent"], "conversation-panel");
assert.equal(headers["X-Initiator"], "user");
assert.ok(headers["x-request-id"]);
});
test("GithubExecutor.buildHeaders forwards valid client x-initiator and falls back for invalid values", () => {
const executor = new GithubExecutor();
const agentHeaders = executor.buildHeaders({ accessToken: "gh-access-token" }, true, {
"x-initiator": "agent",
});
assert.equal(agentHeaders["X-Initiator"], "agent");
const invalidHeaders = executor.buildHeaders({ accessToken: "gh-access-token" }, true, {
"x-initiator": "automation",
});
assert.equal(invalidHeaders["X-Initiator"], "user");
const mixedCaseHeaders = executor.buildHeaders({ accessToken: "gh-access-token" }, true, {
"X-InItIaToR": "agent",
});
assert.equal(mixedCaseHeaders["X-Initiator"], "agent");
});
test("GithubExecutor.execute forwards client x-initiator headers without shared state", async () => {
const executor = new GithubExecutor();
const originalFetch = globalThis.fetch;
const seenInitiators: string[] = [];
globalThis.fetch = async (_url, init: RequestInit = {}) => {
seenInitiators.push((init.headers as Record<string, string>)["X-Initiator"]);
return new Response(JSON.stringify({ choices: [] }), {
status: 200,
headers: { "Content-Type": "application/json" },
});
};
try {
await executor.execute({
model: "gpt-4.1",
body: { messages: [{ role: "user", content: "hi" }] },
stream: false,
credentials: {
accessToken: "gh-access-token",
providerSpecificData: { copilotToken: "copilot-token" },
},
clientHeaders: { "x-initiator": "agent" },
});
await executor.execute({
model: "gpt-4.1",
body: { messages: [{ role: "user", content: "hi" }] },
stream: false,
credentials: {
accessToken: "gh-access-token",
providerSpecificData: { copilotToken: "copilot-token" },
},
clientHeaders: { "x-initiator": "user" },
});
assert.deepEqual(seenInitiators, ["agent", "user"]);
} finally {
globalThis.fetch = originalFetch;
}
});
test("GithubExecutor.refreshCredentials returns Copilot token directly when available", async () => {
const executor = new GithubExecutor();
const originalFetch = globalThis.fetch;
globalThis.fetch = async (url, options) => {
assert.match(String(url), /copilot_internal\/v2\/token$/);
assert.equal(options.headers.Authorization, "token gh-access-token");
return new Response(
JSON.stringify({
token: "copilot-token",
expires_at: 1_777_777_777,
}),
{ status: 200, headers: { "Content-Type": "application/json" } }
);
};
try {
const result = await executor.refreshCredentials({ accessToken: "gh-access-token" }, null);
assert.deepEqual(result, {
accessToken: "gh-access-token",
refreshToken: undefined,
copilotToken: "copilot-token",
copilotTokenExpiresAt: 1_777_777_777,
providerSpecificData: {
copilotToken: "copilot-token",
copilotTokenExpiresAt: 1_777_777_777,
},
});
} finally {
globalThis.fetch = originalFetch;
}
});
test("GithubExecutor.refreshCredentials falls back to GitHub OAuth refresh before retrying Copilot", async () => {
const executor = new GithubExecutor();
const originalFetch = globalThis.fetch;
const calls = [];
globalThis.fetch = async (url, options: RequestInit = {}) => {
calls.push(String(url));
if (String(url).includes("/copilot_internal/v2/token") && calls.length === 1) {
return new Response("unauthorized", { status: 401 });
}
if (String(url).includes("/oauth/access_token")) {
return new Response(
JSON.stringify({
access_token: "new-gh-token",
refresh_token: "new-refresh-token",
expires_in: 3600,
}),
{ status: 200, headers: { "Content-Type": "application/json" } }
);
}
if (String(url).includes("/copilot_internal/v2/token")) {
assert.equal((options.headers as Record<string, string>).Authorization, "token new-gh-token");
return new Response(
JSON.stringify({
token: "new-copilot-token",
expires_at: 1_888_888_888,
}),
{ status: 200, headers: { "Content-Type": "application/json" } }
);
}
throw new Error(`unexpected url: ${url}`);
};
try {
const result = await executor.refreshCredentials(
{
accessToken: "old-gh-token",
refreshToken: "refresh-token",
},
null
);
assert.deepEqual(result, {
accessToken: "new-gh-token",
refreshToken: "new-refresh-token",
expiresIn: 3600,
copilotToken: "new-copilot-token",
copilotTokenExpiresAt: 1_888_888_888,
providerSpecificData: {
copilotToken: "new-copilot-token",
copilotTokenExpiresAt: 1_888_888_888,
},
});
} finally {
globalThis.fetch = originalFetch;
}
});
test("GithubExecutor.needsRefresh checks missing and expiring Copilot tokens", () => {
const executor = new GithubExecutor();
assert.equal(executor.needsRefresh({}), true);
assert.equal(
executor.needsRefresh({
providerSpecificData: {
copilotToken: "copilot-token",
copilotTokenExpiresAt: Math.floor((Date.now() + 60_000) / 1000),
},
}),
true
);
assert.equal(
executor.needsRefresh({
providerSpecificData: {
copilotToken: "copilot-token",
copilotTokenExpiresAt: Math.floor((Date.now() + 60 * 60 * 1000) / 1000),
},
}),
false
);
});
test("GithubExecutor.execute preserves complete SSE responses including terminal [DONE] frames", async () => {
const executor = new GithubExecutor();
const originalFetch = globalThis.fetch;
globalThis.fetch = async () =>
new Response(
new ReadableStream({
start(controller) {
controller.enqueue(new TextEncoder().encode('data: {"chunk":"one"}\n\n'));
controller.enqueue(new TextEncoder().encode("data: [DONE]\n\n"));
controller.close();
},
}),
{
status: 200,
headers: { "Content-Type": "text/event-stream" },
}
);
try {
const result = await executor.execute({
model: "gpt-4.1",
body: { messages: [{ role: "user", content: "hi" }] },
stream: true,
credentials: { accessToken: "gh-access-token" },
});
const text = await result.response.text();
assert.match(text, /"chunk":"one"/);
assert.match(text, /\[DONE\]/);
} finally {
globalThis.fetch = originalFetch;
}
});