172 lines
4.6 KiB
JavaScript
172 lines
4.6 KiB
JavaScript
import createNextIntlPlugin from "next-intl/plugin";
|
|
|
|
const withNextIntl = createNextIntlPlugin("./src/i18n/request.ts");
|
|
|
|
/** @type {import('next').NextConfig} */
|
|
const nextConfig = {
|
|
// Turbopack config: redirect native modules to stubs at build time
|
|
turbopack: {
|
|
resolveAlias: {
|
|
// Point mitm/manager to a stub during build (native child_process/fs can't be bundled)
|
|
"@/mitm/manager": "./src/mitm/manager.stub.ts",
|
|
},
|
|
},
|
|
output: "standalone",
|
|
serverExternalPackages: [
|
|
"pino",
|
|
"pino-pretty",
|
|
"thread-stream",
|
|
"better-sqlite3",
|
|
"keytar",
|
|
"wreq-js",
|
|
"zod",
|
|
"child_process",
|
|
"fs",
|
|
"path",
|
|
"os",
|
|
"crypto",
|
|
"net",
|
|
"tls",
|
|
"http",
|
|
"https",
|
|
"stream",
|
|
"buffer",
|
|
"util",
|
|
],
|
|
transpilePackages: ["@omniroute/open-sse"],
|
|
allowedDevOrigins: ["localhost", "127.0.0.1", "192.168.*"],
|
|
typescript: {
|
|
// TODO: Re-enable after fixing all sub-component useTranslations scope issues
|
|
ignoreBuildErrors: true,
|
|
},
|
|
images: {
|
|
unoptimized: true,
|
|
},
|
|
webpack: (config, { isServer, webpack }) => {
|
|
if (isServer) {
|
|
// Webpack IgnorePlugin: skip thread-stream test files that contain
|
|
// intentionally broken syntax/imports (they cause Turbopack build errors)
|
|
config.plugins.push(
|
|
new webpack.IgnorePlugin({
|
|
resourceRegExp: /\/test\//,
|
|
contextRegExp: /thread-stream/,
|
|
})
|
|
);
|
|
// ── Turbopack / Next.js 16 module-hash patch (#394, #396, #398) ────────
|
|
//
|
|
// Next.js 16 (with or without Turbopack) compiles the instrumentation hook
|
|
// into a separate chunk and emits hashed require() calls such as:
|
|
// require('better-sqlite3-90e2652d1716b047')
|
|
// require('zod-dcb22c6336e0bc69')
|
|
// require('pino-28069d5257187539')
|
|
//
|
|
// These hashed names don't exist in node_modules and cause a 500 at
|
|
// startup on all npm global installs (issues #394, #396, #398).
|
|
//
|
|
// We use two strategies:
|
|
// 1. Exact-name externals for all known server-side packages.
|
|
// 2. Hash-strip catch-all: any require('<name>-<16hexchars>' strips the
|
|
// suffix and falls through to the real package name.
|
|
//
|
|
const HASH_PATTERN = /^(.+)-[0-9a-f]{16}$/;
|
|
|
|
const KNOWN_EXTERNALS = new Set([
|
|
"better-sqlite3",
|
|
"keytar",
|
|
"wreq-js",
|
|
"zod",
|
|
"pino",
|
|
"pino-pretty",
|
|
"child_process",
|
|
"fs",
|
|
"path",
|
|
"os",
|
|
"crypto",
|
|
"net",
|
|
"tls",
|
|
"http",
|
|
"https",
|
|
"stream",
|
|
"buffer",
|
|
"util",
|
|
]);
|
|
|
|
const prev = config.externals ?? [];
|
|
const prevArr = Array.isArray(prev) ? prev : [prev];
|
|
config.externals = [
|
|
...prevArr,
|
|
({ request }, callback) => {
|
|
// Case 1: Exact known package — treat as external
|
|
if (KNOWN_EXTERNALS.has(request)) {
|
|
return callback(null, `commonjs ${request}`);
|
|
}
|
|
// Case 2: Hash-suffixed name — strip hash, use base name
|
|
// e.g. "better-sqlite3-90e2652d1716b047" → "better-sqlite3"
|
|
// "zod-dcb22c6336e0bc69" → "zod"
|
|
const hashMatch = request?.match?.(HASH_PATTERN);
|
|
if (hashMatch) {
|
|
const baseName = hashMatch[1];
|
|
return callback(null, `commonjs ${baseName}`);
|
|
}
|
|
callback();
|
|
},
|
|
];
|
|
} else {
|
|
// Ignore native Node.js modules in browser bundle
|
|
config.resolve.fallback = {
|
|
...config.resolve.fallback,
|
|
fs: false,
|
|
path: false,
|
|
child_process: false,
|
|
net: false,
|
|
tls: false,
|
|
crypto: false,
|
|
};
|
|
}
|
|
return config;
|
|
},
|
|
|
|
async rewrites() {
|
|
return [
|
|
{
|
|
source: "/chat/completions",
|
|
destination: "/api/v1/chat/completions",
|
|
},
|
|
{
|
|
source: "/responses",
|
|
destination: "/api/v1/responses",
|
|
},
|
|
{
|
|
source: "/responses/:path*",
|
|
destination: "/api/v1/responses/:path*",
|
|
},
|
|
{
|
|
source: "/models",
|
|
destination: "/api/v1/models",
|
|
},
|
|
{
|
|
source: "/v1/v1/:path*",
|
|
destination: "/api/v1/:path*",
|
|
},
|
|
{
|
|
source: "/v1/v1",
|
|
destination: "/api/v1",
|
|
},
|
|
{
|
|
source: "/codex/:path*",
|
|
destination: "/api/v1/responses",
|
|
},
|
|
{
|
|
source: "/v1/:path*",
|
|
destination: "/api/v1/:path*",
|
|
},
|
|
{
|
|
source: "/v1",
|
|
destination: "/api/v1",
|
|
},
|
|
];
|
|
},
|
|
};
|
|
|
|
export default withNextIntl(nextConfig);
|