import { type CookieMap, getGoogleCookies } from "./chrome-cookies.js"; const GEMINI_APP_URL = "https://gemini.google.com/app"; const GEMINI_STREAM_GENERATE_URL = "https://gemini.google.com/_/BardChatUi/data/assistant.lamda.BardFrontendService/StreamGenerate"; const GEMINI_UPLOAD_URL = "https://content-push.googleapis.com/upload"; const GEMINI_UPLOAD_PUSH_ID = "feeds/mcudyrk2a4khkz"; const USER_AGENT = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36"; const MODEL_HEADER_NAME = "x-goog-ext-525001261-jspb"; const MODEL_HEADERS: Record = { "gemini-3-pro": '[1,null,null,null,"9d8ca3786ebdfbea",null,null,0,[4]]', "gemini-2.5-pro": '[1,null,null,null,"4af6c7f5da75d65d",null,null,0,[4]]', "gemini-2.5-flash": '[1,null,null,null,"9ec249fc9ad08861",null,null,0,[4]]', }; const REQUIRED_COOKIES = ["__Secure-1PSID", "__Secure-1PSIDTS"]; export interface GeminiWebOptions { youtubeUrl?: string; model?: string; files?: string[]; signal?: AbortSignal; timeoutMs?: number; } function hasRequiredCookies(cookieMap: CookieMap): boolean { return REQUIRED_COOKIES.every((name) => Boolean(cookieMap[name])); } export async function isGeminiWebAvailable(): Promise { const result = await getGoogleCookies(); if (!result || !hasRequiredCookies(result.cookies)) return null; return result.cookies; } export async function queryWithCookies( prompt: string, cookieMap: CookieMap, options: GeminiWebOptions = {}, ): Promise { const model = options.model && MODEL_HEADERS[options.model] ? options.model : "gemini-2.5-flash"; const timeoutMs = options.timeoutMs ?? 120000; let fullPrompt = prompt; if (options.youtubeUrl) { fullPrompt = `${fullPrompt}\n\nYouTube video: ${options.youtubeUrl}`; } const result = await runGeminiWebOnce(fullPrompt, cookieMap, model, options.files, timeoutMs, options.signal); if (isModelUnavailable(result.errorCode) && model !== "gemini-2.5-flash") { const fallback = await runGeminiWebOnce(fullPrompt, cookieMap, "gemini-2.5-flash", options.files, timeoutMs, options.signal); if (fallback.errorMessage) throw new Error(fallback.errorMessage); if (!fallback.text) throw new Error("Gemini Web returned empty response (fallback model)"); return fallback.text; } if (result.errorMessage) throw new Error(result.errorMessage); if (!result.text) throw new Error("Gemini Web returned empty response"); return result.text; } interface GeminiWebResult { text: string; errorCode?: number; errorMessage?: string; } async function runGeminiWebOnce( prompt: string, cookieMap: CookieMap, model: string, files: string[] | undefined, timeoutMs: number, signal?: AbortSignal, ): Promise { const effectiveSignal = withTimeout(signal, timeoutMs); const cookieHeader = buildCookieHeader(cookieMap); const accessToken = await fetchAccessToken(cookieHeader, effectiveSignal); const uploaded: Array<{ id: string; name: string }> = []; if (files) { for (const filePath of files) { uploaded.push(await uploadFile(filePath, cookieHeader, effectiveSignal)); } } const fReq = buildFReqPayload(prompt, uploaded); const params = new URLSearchParams(); params.set("at", accessToken); params.set("f.req", fReq); const res = await fetch(GEMINI_STREAM_GENERATE_URL, { method: "POST", headers: { "content-type": "application/x-www-form-urlencoded;charset=utf-8", host: "gemini.google.com", origin: "https://gemini.google.com", referer: "https://gemini.google.com/", "x-same-domain": "1", "user-agent": USER_AGENT, cookie: cookieHeader, [MODEL_HEADER_NAME]: MODEL_HEADERS[model], }, body: params.toString(), signal: effectiveSignal, }); const rawText = await res.text(); if (!res.ok) { return { text: "", errorMessage: `Gemini request failed: ${res.status}` }; } try { return parseStreamGenerateResponse(rawText); } catch (err) { let errorCode: number | undefined; try { const json = JSON.parse(trimJsonEnvelope(rawText)); errorCode = extractErrorCode(json); } catch {} return { text: "", errorCode, errorMessage: err instanceof Error ? err.message : String(err), }; } } async function fetchAccessToken( cookieHeader: string, signal: AbortSignal, ): Promise { const html = await fetchWithCookieRedirects(GEMINI_APP_URL, cookieHeader, 10, signal); for (const key of ["SNlM0e", "thykhd"]) { const match = html.match(new RegExp(`"${key}":"(.*?)"`)); if (match?.[1]) return match[1]; } throw new Error("Unable to authenticate with Gemini. Make sure you're signed into gemini.google.com in Chrome."); } async function fetchWithCookieRedirects( url: string, cookieHeader: string, maxRedirects: number, signal: AbortSignal, ): Promise { let current = url; for (let i = 0; i <= maxRedirects; i++) { const res = await fetch(current, { headers: { "user-agent": USER_AGENT, cookie: cookieHeader }, redirect: "manual", signal, }); if (res.status >= 300 && res.status < 400) { const location = res.headers.get("location"); if (location) { current = new URL(location, current).toString(); continue; } } return await res.text(); } throw new Error(`Too many redirects (>${maxRedirects})`); } async function uploadFile( filePath: string, cookieHeader: string, signal: AbortSignal, ): Promise<{ id: string; name: string }> { const { readFileSync } = await import("node:fs"); const { basename } = await import("node:path"); const data = readFileSync(filePath); const fileName = basename(filePath); const boundary = "----FormBoundary" + Math.random().toString(36).slice(2); const header = `--${boundary}\r\nContent-Disposition: form-data; name="file"; filename="${fileName}"\r\nContent-Type: application/octet-stream\r\n\r\n`; const footer = `\r\n--${boundary}--\r\n`; const body = Buffer.concat([ Buffer.from(header, "utf-8"), data, Buffer.from(footer, "utf-8"), ]); const res = await fetch(GEMINI_UPLOAD_URL, { method: "POST", headers: { "content-type": `multipart/form-data; boundary=${boundary}`, "push-id": GEMINI_UPLOAD_PUSH_ID, "user-agent": USER_AGENT, cookie: cookieHeader, }, body, signal, }); if (!res.ok) { const text = await res.text(); throw new Error(`File upload failed: ${res.status} (${text.slice(0, 200)})`); } return { id: await res.text(), name: fileName }; } function buildFReqPayload( prompt: string, uploaded: Array<{ id: string; name: string }>, ): string { const promptPayload = uploaded.length > 0 ? [prompt, 0, null, uploaded.map((file) => [[file.id, 1]])] : [prompt]; const innerList = [promptPayload, null, null]; return JSON.stringify([null, JSON.stringify(innerList)]); } function withTimeout(signal: AbortSignal | undefined, timeoutMs: number): AbortSignal { const timeout = AbortSignal.timeout(timeoutMs); return signal ? AbortSignal.any([signal, timeout]) : timeout; } function buildCookieHeader(cookieMap: CookieMap): string { return Object.entries(cookieMap) .filter(([, value]) => typeof value === "string" && value.length > 0) .map(([name, value]) => `${name}=${value}`) .join("; "); } function getNestedValue(value: unknown, pathParts: number[]): unknown { let current: unknown = value; for (const part of pathParts) { if (current == null) return undefined; if (!Array.isArray(current)) return undefined; current = (current as unknown[])[part]; } return current; } function trimJsonEnvelope(text: string): string { const start = text.indexOf("["); const end = text.lastIndexOf("]"); if (start === -1 || end === -1 || end <= start) { throw new Error("Gemini response did not contain a JSON payload."); } return text.slice(start, end + 1); } function extractErrorCode(responseJson: unknown): number | undefined { const code = getNestedValue(responseJson, [0, 5, 2, 0, 1, 0]); return typeof code === "number" && code >= 0 ? code : undefined; } function isModelUnavailable(errorCode: number | undefined): boolean { return errorCode === 1052; } function parseStreamGenerateResponse(rawText: string): GeminiWebResult { const responseJson = JSON.parse(trimJsonEnvelope(rawText)); const errorCode = extractErrorCode(responseJson); const parts = Array.isArray(responseJson) ? responseJson : []; let body: unknown = null; for (let i = 0; i < parts.length; i++) { const partBody = getNestedValue(parts[i], [2]); if (!partBody || typeof partBody !== "string") continue; try { const parsed = JSON.parse(partBody); const candidateList = getNestedValue(parsed, [4]); if (Array.isArray(candidateList) && candidateList.length > 0) { body = parsed; break; } } catch {} } const candidateList = getNestedValue(body, [4]); const firstCandidate = Array.isArray(candidateList) ? (candidateList as unknown[])[0] : undefined; const textRaw = getNestedValue(firstCandidate, [1, 0]) as string | undefined; let text = textRaw ?? ""; if (/^http:\/\/googleusercontent\.com\/card_content\/\d+/.test(text)) { const alt = getNestedValue(firstCandidate, [22, 0]) as string | undefined; if (alt) text = alt; } return { text, errorCode }; }