From 4d93931cfec836548d5d24632333d378b8988483 Mon Sep 17 00:00:00 2001 From: Shu Ding Date: Thu, 18 May 2023 03:31:43 +0200 Subject: [PATCH] Fix custom server React resolution with app dir and pages both presented (#49805) If running Next.js in the custom server with both app and pages directories presented, a standalone server needs to serve all traffic and the server running in main thread should be a no-op. (cc @ijjk). Fixes the problem described here: https://github.com/vercel/next.js/issues/49355#issuecomment-1537536063. --- .../server/lib/render-server-standalone.ts | 4 +- packages/next/src/server/next.ts | 146 ++++++++++++++++-- test/production/custom-server/app/1/page.js | 5 + test/production/custom-server/app/layout.js | 12 ++ .../custom-server/custom-server.test.ts | 13 ++ test/production/custom-server/pages/2.js | 5 + 6 files changed, 167 insertions(+), 18 deletions(-) create mode 100644 test/production/custom-server/app/1/page.js create mode 100644 test/production/custom-server/app/layout.js create mode 100644 test/production/custom-server/pages/2.js diff --git a/packages/next/src/server/lib/render-server-standalone.ts b/packages/next/src/server/lib/render-server-standalone.ts index 83be239a0079c..cb4778b742103 100644 --- a/packages/next/src/server/lib/render-server-standalone.ts +++ b/packages/next/src/server/lib/render-server-standalone.ts @@ -11,11 +11,13 @@ export const createServerHandler = async ({ port, hostname, dir, + dev = false, minimalMode, }: { port: number hostname: string dir: string + dev?: boolean minimalMode: boolean }) => { const routerWorker = new Worker(renderServerPath, { @@ -60,7 +62,7 @@ export const createServerHandler = async ({ const { port: routerPort } = await routerWorker.initialize({ dir, port, - dev: false, + dev, hostname, minimalMode, workerType: 'router', diff --git a/packages/next/src/server/next.ts b/packages/next/src/server/next.ts index 81fbb5f1e1dc1..0a6f8e29a4bbe 100644 --- a/packages/next/src/server/next.ts +++ b/packages/next/src/server/next.ts @@ -2,6 +2,8 @@ import type { Options as DevServerOptions } from './dev/next-dev-server' import type { NodeRequestHandler } from './next-server' import type { UrlWithParsedQuery } from 'url' import type { NextConfigComplete } from './config-shared' +import type { IncomingMessage, ServerResponse } from 'http' +import type { NextParsedUrlQuery, NextUrlWithParsedQuery } from './request-meta' import './require-hook' import './node-polyfill-fetch' @@ -9,14 +11,17 @@ import './node-polyfill-crypto' import { default as Server } from './next-server' import * as log from '../build/output/log' import loadConfig from './config' -import { resolve } from 'path' +import { join, resolve } from 'path' import { NON_STANDARD_NODE_ENV } from '../lib/constants' -import { PHASE_DEVELOPMENT_SERVER } from '../shared/lib/constants' +import { + PHASE_DEVELOPMENT_SERVER, + SERVER_DIRECTORY, +} from '../shared/lib/constants' import { PHASE_PRODUCTION_SERVER } from '../shared/lib/constants' -import { IncomingMessage, ServerResponse } from 'http' -import { NextUrlWithParsedQuery } from './request-meta' import { getTracer } from './lib/trace/tracer' import { NextServerSpan } from './lib/trace/constants' +import { formatUrl } from '../shared/lib/router/utils/format-url' +import { findDir } from '../lib/find-pages-dir' let ServerImpl: typeof Server @@ -29,6 +34,7 @@ const getServerImpl = async () => { export type NextServerOptions = Partial & { preloadedConfig?: NextConfigComplete + internal_setStandaloneConfig?: boolean } export interface RequestHandler { @@ -39,11 +45,17 @@ export interface RequestHandler { ): Promise } +const SYMBOL_SET_STANDALONE_MODE = Symbol('next.set_standalone_mode') +const SYMBOL_LOAD_CONFIG = Symbol('next.load_config') + export class NextServer { private serverPromise?: Promise private server?: Server private reqHandlerPromise?: Promise private preparedAssetPrefix?: string + + private standaloneMode?: boolean + public options: NextServerOptions constructor(options: NextServerOptions) { @@ -58,6 +70,10 @@ export class NextServer { return this.options.port } + [SYMBOL_SET_STANDALONE_MODE]() { + this.standaloneMode = true + } + getRequestHandler(): RequestHandler { return async ( req: IncomingMessage, @@ -127,6 +143,8 @@ export class NextServer { async prepare() { const server = await this.getServer() + if (this.standaloneMode) return + // We shouldn't prepare the server in production, // because this code won't be executed when deployed if (this.options.dev) { @@ -151,7 +169,7 @@ export class NextServer { return server } - private async loadConfig() { + private async [SYMBOL_LOAD_CONFIG]() { return ( this.options.preloadedConfig || loadConfig( @@ -166,7 +184,11 @@ export class NextServer { private async getServer() { if (!this.serverPromise) { - this.serverPromise = this.loadConfig().then(async (conf) => { + 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) { @@ -181,17 +203,6 @@ export class NextServer { } } - if (this.options.customServer !== false) { - // When running as a custom server with app dir, we must set this env - // to correctly alias the React versions. - if (conf.experimental.appDir) { - process.env.__NEXT_PRIVATE_PREBUNDLED_REACT = conf.experimental - .serverActions - ? 'experimental' - : 'next' - } - } - this.server = await this.createServer({ ...this.options, conf, @@ -250,6 +261,107 @@ function createServer(options: NextServerOptions): NextServer { ) } + if (options.customServer !== false) { + // If the `app` dir exists, we'll need to run the standalone server to have + // both types of renderers (pages, app) running in separated processes, + // instead of having the Next server only. + let shouldUseStandaloneMode = false + + const dir = resolve(options.dir || '.') + const server = new NextServer(options) + + const { createServerHandler } = + require('./lib/render-server-standalone') as typeof import('./lib/render-server-standalone') + + let handlerPromise: Promise> + + return new Proxy( + {}, + { + get: function (_, propKey) { + switch (propKey) { + case 'prepare': + return async () => { + // Instead of running Next Server's `prepare`, we'll run the loadConfig first to determine + // if we should run the standalone server or not. + const config = await server[SYMBOL_LOAD_CONFIG]() + + // Check if the application has app dir or not. This depends on the mode (dev or prod). + // For dev, `app` should be existing in the sources and for prod it should be existing + // in the dist folder. + const distDir = + process.env.NEXT_RUNTIME === 'edge' + ? config.distDir + : join(dir, config.distDir) + const serverDistDir = join(distDir, SERVER_DIRECTORY) + const hasAppDir = !!findDir( + options.dev ? dir : serverDistDir, + 'app' + ) + + if (hasAppDir) { + shouldUseStandaloneMode = true + server[SYMBOL_SET_STANDALONE_MODE]() + + handlerPromise = + handlerPromise || + createServerHandler({ + port: options.port || 3000, + dev: options.dev, + dir, + hostname: options.hostname || 'localhost', + minimalMode: false, + }) + } else { + return server.prepare() + } + } + case 'getRequestHandler': { + return () => { + let handler: RequestHandler + return async (req: IncomingMessage, res: ServerResponse) => { + if (shouldUseStandaloneMode) { + const standaloneHandler = await handlerPromise + return standaloneHandler(req, res) + } + handler = handler || server.getRequestHandler() + return handler(req, res) + } + } + } + case 'render': { + return async ( + req: IncomingMessage, + res: ServerResponse, + pathname: string, + query?: NextParsedUrlQuery, + parsedUrl?: NextUrlWithParsedQuery + ) => { + if (shouldUseStandaloneMode) { + const handler = await handlerPromise + req.url = formatUrl({ + ...parsedUrl, + pathname, + query, + }) + return handler(req, res) + } + + return server.render(req, res, pathname, query, parsedUrl) + } + } + default: { + const method = server[propKey as keyof NextServer] + if (typeof method === 'function') { + return method.bind(server) + } + } + } + }, + } + ) as any + } + return new NextServer(options) } diff --git a/test/production/custom-server/app/1/page.js b/test/production/custom-server/app/1/page.js new file mode 100644 index 0000000000000..e9f5bca049b4d --- /dev/null +++ b/test/production/custom-server/app/1/page.js @@ -0,0 +1,5 @@ +import { version } from 'react' + +export default function Page() { + return
app: {version}
+} diff --git a/test/production/custom-server/app/layout.js b/test/production/custom-server/app/layout.js new file mode 100644 index 0000000000000..8525f5f8c0b2a --- /dev/null +++ b/test/production/custom-server/app/layout.js @@ -0,0 +1,12 @@ +export const metadata = { + title: 'Next.js', + description: 'Generated by Next.js', +} + +export default function RootLayout({ children }) { + return ( + + {children} + + ) +} diff --git a/test/production/custom-server/custom-server.test.ts b/test/production/custom-server/custom-server.test.ts index 532f219e4cab0..e7169bf50ee16 100644 --- a/test/production/custom-server/custom-server.test.ts +++ b/test/production/custom-server/custom-server.test.ts @@ -11,5 +11,18 @@ createNextDescribe( const $ = await next.render$(`/${page}`) expect($('p').text()).toBe(`Page ${page}`) }) + + describe('with app dir', () => { + it('should render app with react canary', async () => { + const $ = await next.render$(`/1`) + expect($('body').text()).toMatch(/app: .+-canary/) + }) + + it('should render pages with react stable', async () => { + const $ = await next.render$(`/2`) + expect($('body').text()).toMatch(/pages:/) + expect($('body').text()).not.toMatch(/canary/) + }) + }) } ) diff --git a/test/production/custom-server/pages/2.js b/test/production/custom-server/pages/2.js new file mode 100644 index 0000000000000..532d8319a9a40 --- /dev/null +++ b/test/production/custom-server/pages/2.js @@ -0,0 +1,5 @@ +import { version } from 'react' + +export default function Page() { + return
pages: {version}
+}