mirror of
https://github.com/diegosouzapw/OmniRoute.git
synced 2026-05-23 12:57:09 +00:00
Integrated release/v3.7.2 changes — refreshed web client user agents, env docs, Gemini OAuth fix
285 lines
8.9 KiB
TypeScript
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;
|
|
}
|
|
});
|