177 lines
4.0 KiB
TypeScript
177 lines
4.0 KiB
TypeScript
import { getServerSideConfig } from "@/app/config/server";
|
||
import {
|
||
BAIDU_BASE_URL,
|
||
ApiPath,
|
||
ModelProvider,
|
||
BAIDU_OATUH_URL,
|
||
ServiceProvider,
|
||
} from "@/app/constant";
|
||
import { prettyObject } from "@/app/utils/format";
|
||
import { NextRequest, NextResponse } from "next/server";
|
||
import { auth } from "@/app/api/auth";
|
||
import { isModelAvailableInServer } from "@/app/utils/model";
|
||
|
||
const serverConfig = getServerSideConfig();
|
||
|
||
async function handle(
|
||
req: NextRequest,
|
||
{ params }: { params: { path: string[] } },
|
||
) {
|
||
console.log("[Baidu Route] params ", params);
|
||
|
||
if (req.method === "OPTIONS") {
|
||
return NextResponse.json({ body: "OK" }, { status: 200 });
|
||
}
|
||
|
||
const authResult = auth(req, ModelProvider.Ernie);
|
||
if (authResult.error) {
|
||
return NextResponse.json(authResult, {
|
||
status: 401,
|
||
});
|
||
}
|
||
|
||
try {
|
||
const response = await request(req);
|
||
return response;
|
||
} catch (e) {
|
||
console.error("[Baidu] ", e);
|
||
return NextResponse.json(prettyObject(e));
|
||
}
|
||
}
|
||
|
||
export const GET = handle;
|
||
export const POST = handle;
|
||
|
||
export const runtime = "edge";
|
||
export const preferredRegion = [
|
||
"arn1",
|
||
"bom1",
|
||
"cdg1",
|
||
"cle1",
|
||
"cpt1",
|
||
"dub1",
|
||
"fra1",
|
||
"gru1",
|
||
"hnd1",
|
||
"iad1",
|
||
"icn1",
|
||
"kix1",
|
||
"lhr1",
|
||
"pdx1",
|
||
"sfo1",
|
||
"sin1",
|
||
"syd1",
|
||
];
|
||
|
||
async function request(req: NextRequest) {
|
||
const controller = new AbortController();
|
||
|
||
let path = `${req.nextUrl.pathname}`.replaceAll(ApiPath.Baidu, "");
|
||
|
||
let baseUrl = serverConfig.baiduUrl || BAIDU_BASE_URL;
|
||
|
||
if (!baseUrl.startsWith("http")) {
|
||
baseUrl = `https://${baseUrl}`;
|
||
}
|
||
|
||
if (baseUrl.endsWith("/")) {
|
||
baseUrl = baseUrl.slice(0, -1);
|
||
}
|
||
|
||
console.log("[Proxy] ", path);
|
||
console.log("[Base Url]", baseUrl);
|
||
|
||
const timeoutId = setTimeout(
|
||
() => {
|
||
controller.abort();
|
||
},
|
||
10 * 60 * 1000,
|
||
);
|
||
|
||
const { access_token } = await getAccessToken();
|
||
const fetchUrl = `${baseUrl}${path}?access_token=${access_token}`;
|
||
|
||
const fetchOptions: RequestInit = {
|
||
headers: {
|
||
"Content-Type": "application/json",
|
||
},
|
||
method: req.method,
|
||
body: req.body,
|
||
redirect: "manual",
|
||
// @ts-ignore
|
||
duplex: "half",
|
||
signal: controller.signal,
|
||
};
|
||
|
||
// #1815 try to refuse some request to some models
|
||
if (serverConfig.customModels && req.body) {
|
||
try {
|
||
const clonedBody = await req.text();
|
||
fetchOptions.body = clonedBody;
|
||
|
||
const jsonBody = JSON.parse(clonedBody) as { model?: string };
|
||
|
||
// not undefined and is false
|
||
if (
|
||
isModelAvailableInServer(
|
||
serverConfig.customModels,
|
||
jsonBody?.model as string,
|
||
ServiceProvider.Baidu as string,
|
||
)
|
||
) {
|
||
return NextResponse.json(
|
||
{
|
||
error: true,
|
||
message: `you are not allowed to use ${jsonBody?.model} model`,
|
||
},
|
||
{
|
||
status: 403,
|
||
},
|
||
);
|
||
}
|
||
} catch (e) {
|
||
console.error(`[Baidu] filter`, e);
|
||
}
|
||
}
|
||
console.log("[Baidu request]", fetchOptions.headers, req.method);
|
||
try {
|
||
const res = await fetch(fetchUrl, fetchOptions);
|
||
|
||
console.log("[Baidu response]", res.status, " ", res.headers, res.url);
|
||
// to prevent browser prompt for credentials
|
||
const newHeaders = new Headers(res.headers);
|
||
newHeaders.delete("www-authenticate");
|
||
// to disable nginx buffering
|
||
newHeaders.set("X-Accel-Buffering", "no");
|
||
|
||
return new Response(res.body, {
|
||
status: res.status,
|
||
statusText: res.statusText,
|
||
headers: newHeaders,
|
||
});
|
||
} finally {
|
||
clearTimeout(timeoutId);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 使用 AK,SK 生成鉴权签名(Access Token)
|
||
* @return 鉴权签名信息
|
||
*/
|
||
async function getAccessToken(): Promise<{
|
||
access_token: string;
|
||
expires_in: number;
|
||
error?: number;
|
||
}> {
|
||
const AK = serverConfig.baiduApiKey;
|
||
const SK = serverConfig.baiduSecretKey;
|
||
const res = await fetch(
|
||
`${BAIDU_OATUH_URL}?grant_type=client_credentials&client_id=${AK}&client_secret=${SK}`,
|
||
{
|
||
method: "POST",
|
||
},
|
||
);
|
||
const resJson = await res.json();
|
||
return resJson;
|
||
}
|