2023-06-06 17:02:01 +00:00
|
|
|
import { NextRequest, NextResponse } from "next/server";
|
2023-11-08 19:01:29 +00:00
|
|
|
import { getServerSideConfig } from "../config/server";
|
2023-12-23 18:15:30 +00:00
|
|
|
import { DEFAULT_MODELS, OPENAI_BASE_URL, GEMINI_BASE_URL } from "../constant";
|
2023-11-09 18:43:30 +00:00
|
|
|
import { collectModelTable } from "../utils/model";
|
|
|
|
import { makeAzurePath } from "../azure";
|
2023-03-29 17:45:26 +00:00
|
|
|
|
2023-11-08 19:01:29 +00:00
|
|
|
const serverConfig = getServerSideConfig();
|
2023-03-29 17:45:26 +00:00
|
|
|
|
|
|
|
export async function requestOpenai(req: NextRequest) {
|
2023-05-18 08:52:32 +00:00
|
|
|
const controller = new AbortController();
|
2023-11-09 18:43:30 +00:00
|
|
|
|
2023-12-25 02:25:43 +00:00
|
|
|
var authValue,
|
|
|
|
authHeaderName = "";
|
2023-12-25 01:56:51 +00:00
|
|
|
if (serverConfig.isAzure) {
|
2023-12-25 02:25:43 +00:00
|
|
|
authValue =
|
2023-12-25 01:56:51 +00:00
|
|
|
req.headers
|
|
|
|
.get("Authorization")
|
|
|
|
?.trim()
|
|
|
|
.replaceAll("Bearer ", "")
|
|
|
|
.trim() ?? "";
|
2023-12-25 02:25:43 +00:00
|
|
|
|
|
|
|
authHeaderName = "api-key";
|
2023-12-25 01:56:51 +00:00
|
|
|
} else {
|
2023-12-25 02:25:43 +00:00
|
|
|
authValue = req.headers.get("Authorization") ?? "";
|
|
|
|
authHeaderName = "Authorization";
|
2023-12-25 01:56:51 +00:00
|
|
|
}
|
2023-11-09 18:43:30 +00:00
|
|
|
|
|
|
|
let path = `${req.nextUrl.pathname}${req.nextUrl.search}`.replaceAll(
|
2023-05-03 15:08:37 +00:00
|
|
|
"/api/openai/",
|
2023-05-18 08:55:51 +00:00
|
|
|
"",
|
2023-05-03 15:08:37 +00:00
|
|
|
);
|
2023-03-29 17:45:26 +00:00
|
|
|
|
2023-11-09 18:43:30 +00:00
|
|
|
let baseUrl =
|
2023-11-10 07:44:07 +00:00
|
|
|
serverConfig.azureUrl || serverConfig.baseUrl || OPENAI_BASE_URL;
|
2023-04-14 18:50:04 +00:00
|
|
|
|
|
|
|
if (!baseUrl.startsWith("http")) {
|
2023-11-08 19:01:29 +00:00
|
|
|
baseUrl = `https://${baseUrl}`;
|
2023-04-14 18:50:04 +00:00
|
|
|
}
|
|
|
|
|
2023-11-07 23:09:52 +00:00
|
|
|
if (baseUrl.endsWith("/")) {
|
2023-08-10 02:47:06 +00:00
|
|
|
baseUrl = baseUrl.slice(0, -1);
|
|
|
|
}
|
|
|
|
|
2023-11-09 18:43:30 +00:00
|
|
|
console.log("[Proxy] ", path);
|
2023-04-14 18:50:04 +00:00
|
|
|
console.log("[Base Url]", baseUrl);
|
2023-11-25 03:03:41 +00:00
|
|
|
// this fix [Org ID] undefined in server side if not using custom point
|
2023-11-25 03:23:16 +00:00
|
|
|
if (serverConfig.openaiOrgId !== undefined) {
|
2023-11-25 03:03:41 +00:00
|
|
|
console.log("[Org ID]", serverConfig.openaiOrgId);
|
|
|
|
}
|
2023-04-19 11:28:33 +00:00
|
|
|
|
2023-11-07 23:09:52 +00:00
|
|
|
const timeoutId = setTimeout(
|
|
|
|
() => {
|
|
|
|
controller.abort();
|
|
|
|
},
|
|
|
|
10 * 60 * 1000,
|
|
|
|
);
|
2023-05-18 08:52:32 +00:00
|
|
|
|
2023-11-09 18:43:30 +00:00
|
|
|
if (serverConfig.isAzure) {
|
|
|
|
if (!serverConfig.azureApiVersion) {
|
|
|
|
return NextResponse.json({
|
|
|
|
error: true,
|
|
|
|
message: `missing AZURE_API_VERSION in server env vars`,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
path = makeAzurePath(path, serverConfig.azureApiVersion);
|
|
|
|
}
|
|
|
|
|
|
|
|
const fetchUrl = `${baseUrl}/${path}`;
|
2023-05-20 16:06:28 +00:00
|
|
|
const fetchOptions: RequestInit = {
|
|
|
|
headers: {
|
|
|
|
"Content-Type": "application/json",
|
2023-07-11 07:46:40 +00:00
|
|
|
"Cache-Control": "no-store",
|
2023-11-09 18:43:30 +00:00
|
|
|
[authHeaderName]: authValue,
|
|
|
|
...(serverConfig.openaiOrgId && {
|
|
|
|
"OpenAI-Organization": serverConfig.openaiOrgId,
|
2023-05-20 16:06:28 +00:00
|
|
|
}),
|
|
|
|
},
|
|
|
|
method: req.method,
|
2023-06-06 18:24:45 +00:00
|
|
|
body: req.body,
|
2023-08-08 13:36:37 +00:00
|
|
|
// to fix #2485: https://stackoverflow.com/questions/55920957/cloudflare-worker-typeerror-one-time-use-body
|
|
|
|
redirect: "manual",
|
2023-06-20 03:57:31 +00:00
|
|
|
// @ts-ignore
|
|
|
|
duplex: "half",
|
2023-05-20 16:06:28 +00:00
|
|
|
signal: controller.signal,
|
|
|
|
};
|
|
|
|
|
2023-06-06 17:02:01 +00:00
|
|
|
// #1815 try to refuse gpt4 request
|
2023-11-08 19:01:29 +00:00
|
|
|
if (serverConfig.customModels && req.body) {
|
2023-06-06 17:02:01 +00:00
|
|
|
try {
|
2023-11-08 19:01:29 +00:00
|
|
|
const modelTable = collectModelTable(
|
|
|
|
DEFAULT_MODELS,
|
|
|
|
serverConfig.customModels,
|
|
|
|
);
|
2023-06-06 18:24:45 +00:00
|
|
|
const clonedBody = await req.text();
|
|
|
|
fetchOptions.body = clonedBody;
|
|
|
|
|
2023-11-08 19:01:29 +00:00
|
|
|
const jsonBody = JSON.parse(clonedBody) as { model?: string };
|
2023-06-06 18:24:45 +00:00
|
|
|
|
2023-11-08 19:01:29 +00:00
|
|
|
// not undefined and is false
|
2023-11-11 16:46:21 +00:00
|
|
|
if (modelTable[jsonBody?.model ?? ""].available === false) {
|
2023-06-06 17:02:01 +00:00
|
|
|
return NextResponse.json(
|
|
|
|
{
|
|
|
|
error: true,
|
2023-11-08 19:01:29 +00:00
|
|
|
message: `you are not allowed to use ${jsonBody?.model} model`,
|
2023-06-06 17:02:01 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
status: 403,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
console.error("[OpenAI] gpt4 filter", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-18 08:52:32 +00:00
|
|
|
try {
|
2023-05-20 16:06:28 +00:00
|
|
|
const res = await fetch(fetchUrl, fetchOptions);
|
|
|
|
|
2023-06-08 15:49:06 +00:00
|
|
|
// to prevent browser prompt for credentials
|
|
|
|
const newHeaders = new Headers(res.headers);
|
|
|
|
newHeaders.delete("www-authenticate");
|
2023-07-10 02:09:19 +00:00
|
|
|
// to disable nginx buffering
|
2023-06-08 15:49:06 +00:00
|
|
|
newHeaders.set("X-Accel-Buffering", "no");
|
2023-05-20 16:06:28 +00:00
|
|
|
|
2023-11-10 12:38:53 +00:00
|
|
|
// The latest version of the OpenAI API forced the content-encoding to be "br" in json response
|
|
|
|
// So if the streaming is disabled, we need to remove the content-encoding header
|
2023-11-10 12:42:12 +00:00
|
|
|
// Because Vercel uses gzip to compress the response, if we don't remove the content-encoding header
|
|
|
|
// The browser will try to decode the response with brotli and fail
|
2023-11-10 12:38:53 +00:00
|
|
|
newHeaders.delete("content-encoding");
|
|
|
|
|
2023-06-08 15:49:06 +00:00
|
|
|
return new Response(res.body, {
|
|
|
|
status: res.status,
|
|
|
|
statusText: res.statusText,
|
|
|
|
headers: newHeaders,
|
|
|
|
});
|
2023-05-18 08:52:32 +00:00
|
|
|
} finally {
|
|
|
|
clearTimeout(timeoutId);
|
|
|
|
}
|
2023-03-29 17:45:26 +00:00
|
|
|
}
|