264 lines
9.7 KiB
JavaScript
264 lines
9.7 KiB
JavaScript
|
import "./require-hook";
|
||
|
import "./node-polyfill-fetch";
|
||
|
import "./node-polyfill-crypto";
|
||
|
import * as log from "../build/output/log";
|
||
|
import loadConfig from "./config";
|
||
|
import path, { resolve } from "path";
|
||
|
import { NON_STANDARD_NODE_ENV } from "../lib/constants";
|
||
|
import { PHASE_DEVELOPMENT_SERVER, SERVER_FILES_MANIFEST } from "../shared/lib/constants";
|
||
|
import { PHASE_PRODUCTION_SERVER } from "../shared/lib/constants";
|
||
|
import { getTracer } from "./lib/trace/tracer";
|
||
|
import { NextServerSpan } from "./lib/trace/constants";
|
||
|
import { formatUrl } from "../shared/lib/router/utils/format-url";
|
||
|
import { checkIsNodeDebugging } from "./lib/is-node-debugging";
|
||
|
let ServerImpl;
|
||
|
const getServerImpl = async ()=>{
|
||
|
if (ServerImpl === undefined) {
|
||
|
ServerImpl = (await Promise.resolve(require("./next-server"))).default;
|
||
|
}
|
||
|
return ServerImpl;
|
||
|
};
|
||
|
const SYMBOL_LOAD_CONFIG = Symbol("next.load_config");
|
||
|
export class NextServer {
|
||
|
constructor(options){
|
||
|
this.options = options;
|
||
|
}
|
||
|
get hostname() {
|
||
|
return this.options.hostname;
|
||
|
}
|
||
|
get port() {
|
||
|
return this.options.port;
|
||
|
}
|
||
|
getRequestHandler() {
|
||
|
return async (req, res, parsedUrl)=>{
|
||
|
return getTracer().trace(NextServerSpan.getRequestHandler, async ()=>{
|
||
|
const requestHandler = await this.getServerRequestHandler();
|
||
|
return requestHandler(req, res, parsedUrl);
|
||
|
});
|
||
|
};
|
||
|
}
|
||
|
getUpgradeHandler() {
|
||
|
return async (req, socket, head)=>{
|
||
|
const server = await this.getServer();
|
||
|
// @ts-expect-error we mark this as protected so it
|
||
|
// causes an error here
|
||
|
return server.handleUpgrade.apply(server, [
|
||
|
req,
|
||
|
socket,
|
||
|
head
|
||
|
]);
|
||
|
};
|
||
|
}
|
||
|
setAssetPrefix(assetPrefix) {
|
||
|
if (this.server) {
|
||
|
this.server.setAssetPrefix(assetPrefix);
|
||
|
} else {
|
||
|
this.preparedAssetPrefix = assetPrefix;
|
||
|
}
|
||
|
}
|
||
|
logError(...args) {
|
||
|
if (this.server) {
|
||
|
this.server.logError(...args);
|
||
|
}
|
||
|
}
|
||
|
async render(...args) {
|
||
|
const server = await this.getServer();
|
||
|
return server.render(...args);
|
||
|
}
|
||
|
async renderToHTML(...args) {
|
||
|
const server = await this.getServer();
|
||
|
return server.renderToHTML(...args);
|
||
|
}
|
||
|
async renderError(...args) {
|
||
|
const server = await this.getServer();
|
||
|
return server.renderError(...args);
|
||
|
}
|
||
|
async renderErrorToHTML(...args) {
|
||
|
const server = await this.getServer();
|
||
|
return server.renderErrorToHTML(...args);
|
||
|
}
|
||
|
async render404(...args) {
|
||
|
const server = await this.getServer();
|
||
|
return server.render404(...args);
|
||
|
}
|
||
|
async prepare(serverFields) {
|
||
|
if (this.standaloneMode) return;
|
||
|
const server = await this.getServer();
|
||
|
if (serverFields) {
|
||
|
Object.assign(server, serverFields);
|
||
|
}
|
||
|
// We shouldn't prepare the server in production,
|
||
|
// because this code won't be executed when deployed
|
||
|
if (this.options.dev) {
|
||
|
await server.prepare();
|
||
|
}
|
||
|
}
|
||
|
async close() {
|
||
|
const server = await this.getServer();
|
||
|
return server.close();
|
||
|
}
|
||
|
async createServer(options) {
|
||
|
let ServerImplementation;
|
||
|
if (options.dev) {
|
||
|
ServerImplementation = require("./dev/next-dev-server").default;
|
||
|
} else {
|
||
|
ServerImplementation = await getServerImpl();
|
||
|
}
|
||
|
const server = new ServerImplementation(options);
|
||
|
return server;
|
||
|
}
|
||
|
async [SYMBOL_LOAD_CONFIG]() {
|
||
|
const dir = resolve(this.options.dir || ".");
|
||
|
const config = this.options.preloadedConfig || await loadConfig(this.options.dev ? PHASE_DEVELOPMENT_SERVER : PHASE_PRODUCTION_SERVER, dir, {
|
||
|
customConfig: this.options.conf,
|
||
|
silent: true
|
||
|
});
|
||
|
// check serialized build config when available
|
||
|
if (process.env.NODE_ENV === "production") {
|
||
|
try {
|
||
|
const serializedConfig = require(path.join(dir, ".next", SERVER_FILES_MANIFEST)).config;
|
||
|
// @ts-expect-error internal field
|
||
|
config.experimental.isExperimentalCompile = serializedConfig.experimental.isExperimentalCompile;
|
||
|
} catch (_) {
|
||
|
// if distDir is customized we don't know until we
|
||
|
// load the config so fallback to loading the config
|
||
|
// from next.config.js
|
||
|
}
|
||
|
}
|
||
|
return config;
|
||
|
}
|
||
|
async getServer() {
|
||
|
if (!this.serverPromise) {
|
||
|
this.serverPromise = this[SYMBOL_LOAD_CONFIG]().then(async (conf)=>{
|
||
|
if (this.standaloneMode) {
|
||
|
process.env.__NEXT_PRIVATE_STANDALONE_CONFIG = JSON.stringify(conf);
|
||
|
}
|
||
|
if (!this.options.dev) {
|
||
|
if (conf.output === "standalone") {
|
||
|
if (!process.env.__NEXT_PRIVATE_STANDALONE_CONFIG) {
|
||
|
log.warn(`"next start" does not work with "output: standalone" configuration. Use "node .next/standalone/server.js" instead.`);
|
||
|
}
|
||
|
} else if (conf.output === "export") {
|
||
|
throw new Error(`"next start" does not work with "output: export" configuration. Use "npx serve@latest out" instead.`);
|
||
|
}
|
||
|
}
|
||
|
this.server = await this.createServer({
|
||
|
...this.options,
|
||
|
conf
|
||
|
});
|
||
|
if (this.preparedAssetPrefix) {
|
||
|
this.server.setAssetPrefix(this.preparedAssetPrefix);
|
||
|
}
|
||
|
return this.server;
|
||
|
});
|
||
|
}
|
||
|
return this.serverPromise;
|
||
|
}
|
||
|
async getServerRequestHandler() {
|
||
|
if (this.reqHandler) return this.reqHandler;
|
||
|
// Memoize request handler creation
|
||
|
if (!this.reqHandlerPromise) {
|
||
|
this.reqHandlerPromise = this.getServer().then((server)=>{
|
||
|
this.reqHandler = getTracer().wrap(NextServerSpan.getServerRequestHandler, server.getRequestHandler().bind(server));
|
||
|
delete this.reqHandlerPromise;
|
||
|
return this.reqHandler;
|
||
|
});
|
||
|
}
|
||
|
return this.reqHandlerPromise;
|
||
|
}
|
||
|
}
|
||
|
class NextCustomServer extends NextServer {
|
||
|
async prepare() {
|
||
|
const { getRequestHandlers } = require("./lib/start-server");
|
||
|
const isNodeDebugging = checkIsNodeDebugging();
|
||
|
const initResult = await getRequestHandlers({
|
||
|
dir: this.options.dir,
|
||
|
port: this.options.port || 3000,
|
||
|
isDev: !!this.options.dev,
|
||
|
hostname: this.options.hostname || "localhost",
|
||
|
minimalMode: this.options.minimalMode,
|
||
|
isNodeDebugging: !!isNodeDebugging
|
||
|
});
|
||
|
this.requestHandler = initResult[0];
|
||
|
this.upgradeHandler = initResult[1];
|
||
|
}
|
||
|
setupWebSocketHandler(customServer, _req) {
|
||
|
if (!this.didWebSocketSetup) {
|
||
|
var _this;
|
||
|
this.didWebSocketSetup = true;
|
||
|
customServer = customServer || ((_this = _req == null ? void 0 : _req.socket) == null ? void 0 : _this.server);
|
||
|
if (customServer) {
|
||
|
customServer.on("upgrade", async (req, socket, head)=>{
|
||
|
this.upgradeHandler(req, socket, head);
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
getRequestHandler() {
|
||
|
return async (req, res, parsedUrl)=>{
|
||
|
this.setupWebSocketHandler(this.options.httpServer, req);
|
||
|
if (parsedUrl) {
|
||
|
req.url = formatUrl(parsedUrl);
|
||
|
}
|
||
|
return this.requestHandler(req, res);
|
||
|
};
|
||
|
}
|
||
|
async render(...args) {
|
||
|
let [req, res, pathname, query, parsedUrl] = args;
|
||
|
this.setupWebSocketHandler(this.options.httpServer, req);
|
||
|
if (!pathname.startsWith("/")) {
|
||
|
console.error(`Cannot render page with path "${pathname}"`);
|
||
|
pathname = `/${pathname}`;
|
||
|
}
|
||
|
pathname = pathname === "/index" ? "/" : pathname;
|
||
|
req.url = formatUrl({
|
||
|
...parsedUrl,
|
||
|
pathname,
|
||
|
query
|
||
|
});
|
||
|
await this.requestHandler(req, res);
|
||
|
return;
|
||
|
}
|
||
|
constructor(...args){
|
||
|
super(...args);
|
||
|
this.standaloneMode = true;
|
||
|
this.didWebSocketSetup = false;
|
||
|
}
|
||
|
}
|
||
|
// This file is used for when users run `require('next')`
|
||
|
function createServer(options) {
|
||
|
// The package is used as a TypeScript plugin.
|
||
|
if (options && "typescript" in options && "version" in options.typescript) {
|
||
|
return require("./next-typescript").createTSPlugin(options);
|
||
|
}
|
||
|
if (options == null) {
|
||
|
throw new Error("The server has not been instantiated properly. https://nextjs.org/docs/messages/invalid-server-options");
|
||
|
}
|
||
|
if (!("isNextDevCommand" in options) && process.env.NODE_ENV && ![
|
||
|
"production",
|
||
|
"development",
|
||
|
"test"
|
||
|
].includes(process.env.NODE_ENV)) {
|
||
|
log.warn(NON_STANDARD_NODE_ENV);
|
||
|
}
|
||
|
if (options.dev && typeof options.dev !== "boolean") {
|
||
|
console.warn("Warning: 'dev' is not a boolean which could introduce unexpected behavior. https://nextjs.org/docs/messages/invalid-server-options");
|
||
|
}
|
||
|
// When the caller is a custom server (using next()).
|
||
|
if (options.customServer !== false) {
|
||
|
const dir = resolve(options.dir || ".");
|
||
|
return new NextCustomServer({
|
||
|
...options,
|
||
|
dir
|
||
|
});
|
||
|
}
|
||
|
// When the caller is Next.js internals (i.e. render worker, start server, etc)
|
||
|
return new NextServer(options);
|
||
|
}
|
||
|
// Support commonjs `require('next')`
|
||
|
module.exports = createServer;
|
||
|
// exports = module.exports
|
||
|
// Support `import next from 'next'`
|
||
|
export default createServer;
|
||
|
|
||
|
//# sourceMappingURL=next.js.map
|