vite-plugin-react-server 1.3.6 → 1.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +32 -18
- package/dist/package.json +4 -2
- package/dist/plugin/config/resolveOptions.d.ts.map +1 -1
- package/dist/plugin/config/resolveOptions.js +4 -2
- package/dist/plugin/dev-server/plugin.client.d.ts.map +1 -1
- package/dist/plugin/dev-server/plugin.client.js +13 -3
- package/dist/plugin/dev-server/plugin.server.d.ts.map +1 -1
- package/dist/plugin/dev-server/plugin.server.js +54 -4
- package/dist/plugin/react-static/plugin.server.d.ts.map +1 -1
- package/dist/plugin/react-static/plugin.server.js +9 -1
- package/dist/plugin/react-static/renderPagesBatched.d.ts.map +1 -1
- package/dist/plugin/react-static/renderPagesBatched.js +136 -36
- package/dist/plugin/react-static/types.d.ts +1 -0
- package/dist/plugin/react-static/types.d.ts.map +1 -1
- package/dist/plugin/types.d.ts +15 -0
- package/dist/plugin/types.d.ts.map +1 -1
- package/dist/plugin/utils/createReactFetcher.js +24 -2
- package/dist/plugin/utils/useRscHmr.js +10 -2
- package/dist/plugin/vendor/register-vendor.js +1 -1
- package/dist/plugin/vendor/vendor-alias.d.ts +5 -2
- package/dist/plugin/vendor/vendor-alias.d.ts.map +1 -1
- package/dist/plugin/vendor/vendor-alias.js +60 -26
- package/dist/plugin/vendor/vendor.server.d.ts.map +1 -1
- package/dist/plugin/vendor/vendor.server.js +2 -2
- package/dist/plugin/vendor/vendor.static.d.ts.map +1 -1
- package/dist/plugin/vendor/vendor.static.js +2 -2
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-client.browser.development.js +1 -1
- package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-client.browser.production.js +1 -1
- package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-client.node.development.js +1 -1
- package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-client.node.production.js +1 -1
- package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-server.node.development.js +1 -1
- package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-server.node.production.js +1 -1
- package/oss-experimental/react-server-dom-esm/esm/react-server-dom-esm-client.browser.development.js +1 -1
- package/oss-experimental/react-server-dom-esm/esm/react-server-dom-esm-client.browser.production.js +1 -1
- package/oss-experimental/react-server-dom-esm/package.json +3 -3
- package/package.json +4 -2
- package/plugin/config/resolveOptions.ts +2 -0
- package/plugin/dev-server/plugin.client.ts +13 -2
- package/plugin/dev-server/plugin.server.ts +71 -4
- package/plugin/react-static/plugin.server.ts +11 -1
- package/plugin/react-static/renderPagesBatched.ts +148 -39
- package/plugin/react-static/types.ts +1 -0
- package/plugin/types.ts +15 -0
- package/plugin/vendor/register-vendor.ts +1 -1
- package/plugin/vendor/vendor-alias.ts +61 -39
- package/plugin/vendor/vendor.server.ts +3 -3
- package/plugin/vendor/vendor.static.ts +3 -2
|
@@ -5,6 +5,7 @@
|
|
|
5
5
|
*/
|
|
6
6
|
import { configureReactServer } from './configureReactServer.server.js';
|
|
7
7
|
import { resolveOptions } from '../config/resolveOptions.js';
|
|
8
|
+
import { readFileSync } from 'node:fs';
|
|
8
9
|
|
|
9
10
|
const vitePluginReactDevServer = function _vitePluginReactServerDevServer(options) {
|
|
10
11
|
if (options == null) {
|
|
@@ -22,13 +23,59 @@ const vitePluginReactDevServer = function _vitePluginReactServerDevServer(option
|
|
|
22
23
|
name: "vite-plugin-react-server:server-hmr",
|
|
23
24
|
apply: "serve",
|
|
24
25
|
// Server-level handleHotUpdate — sends custom WS event to client
|
|
26
|
+
// Vite 6 Environment API: hotUpdate runs per-environment.
|
|
27
|
+
// Prevent server/ssr environments from triggering page reload for client components.
|
|
28
|
+
hotUpdate(ctx) {
|
|
29
|
+
const { file, server } = ctx;
|
|
30
|
+
const envName = ctx.environment?.name ?? "unknown";
|
|
31
|
+
const moduleBase = userOptions.moduleBase || "src";
|
|
32
|
+
const projectRoot = userOptions.projectRoot || server?.config?.root || "";
|
|
33
|
+
const normalizedFile = file.replace(projectRoot, "").replace(/^\/+/, "");
|
|
34
|
+
const isSourceFile = normalizedFile.startsWith(moduleBase + "/");
|
|
35
|
+
if (!isSourceFile) return;
|
|
36
|
+
if (envName === "client") {
|
|
37
|
+
const isClient = (file.endsWith(".tsx") || file.endsWith(".ts") || file.endsWith(".jsx") || file.endsWith(".js")) && (() => {
|
|
38
|
+
try {
|
|
39
|
+
const head = readFileSync(file, "utf-8").slice(0, 200);
|
|
40
|
+
return /^\s*["']use client["']/.test(head.split("\n")[0]);
|
|
41
|
+
} catch {
|
|
42
|
+
return false;
|
|
43
|
+
}
|
|
44
|
+
})();
|
|
45
|
+
if (isClient) return;
|
|
46
|
+
server.config.logger.info(`[vite-plugin-react-server] File changed (RSC refetch): ${normalizedFile}`);
|
|
47
|
+
server.ws.send({
|
|
48
|
+
type: "custom",
|
|
49
|
+
event: "vite-plugin-react-server:server-component-update",
|
|
50
|
+
data: { file: normalizedFile, path: file }
|
|
51
|
+
});
|
|
52
|
+
return [];
|
|
53
|
+
}
|
|
54
|
+
if (envName === "server") {
|
|
55
|
+
const mod = ctx.environment?.moduleGraph?.getModulesByFile(file);
|
|
56
|
+
if (mod) {
|
|
57
|
+
for (const m of mod) {
|
|
58
|
+
ctx.environment.moduleGraph.invalidateModule(m);
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
return [];
|
|
63
|
+
},
|
|
25
64
|
handleHotUpdate({ file, server }) {
|
|
26
65
|
const moduleBase = userOptions.moduleBase || "src";
|
|
27
66
|
const projectRoot = userOptions.projectRoot || server.config.root;
|
|
28
67
|
const normalizedFile = file.replace(projectRoot, "").replace(/^\/+/, "");
|
|
29
|
-
const
|
|
30
|
-
|
|
31
|
-
|
|
68
|
+
const isSourceFile = normalizedFile.startsWith(moduleBase + "/") && (file.endsWith(".tsx") || file.endsWith(".ts") || file.endsWith(".jsx") || file.endsWith(".js"));
|
|
69
|
+
const isClientFile = isSourceFile && (() => {
|
|
70
|
+
try {
|
|
71
|
+
const head = readFileSync(file, "utf-8").slice(0, 200);
|
|
72
|
+
return /^\s*["']use client["']/.test(head.split("\n")[0]);
|
|
73
|
+
} catch {
|
|
74
|
+
return false;
|
|
75
|
+
}
|
|
76
|
+
})();
|
|
77
|
+
if (isSourceFile && !isClientFile) {
|
|
78
|
+
server.config.logger.info(`[vite-plugin-react-server] File changed (RSC refetch): ${normalizedFile}`);
|
|
32
79
|
server.ws.send({
|
|
33
80
|
type: "custom",
|
|
34
81
|
event: "vite-plugin-react-server:server-component-update",
|
|
@@ -45,6 +92,9 @@ const vitePluginReactDevServer = function _vitePluginReactServerDevServer(option
|
|
|
45
92
|
}
|
|
46
93
|
return [];
|
|
47
94
|
}
|
|
95
|
+
if (isClientFile) {
|
|
96
|
+
return [];
|
|
97
|
+
}
|
|
48
98
|
}
|
|
49
99
|
};
|
|
50
100
|
const serverPlugin = {
|
|
@@ -105,4 +155,4 @@ const vitePluginReactDevServer = function _vitePluginReactServerDevServer(option
|
|
|
105
155
|
};
|
|
106
156
|
|
|
107
157
|
export { vitePluginReactDevServer };
|
|
108
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"plugin.server.js","sources":["../../../plugin/dev-server/plugin.server.ts"],"sourcesContent":["import type { StreamPluginOptions } from \"../../types.js\";\nimport { configureReactServer } from \"./configureReactServer.server.js\";\nimport { resolveOptions } from \"../config/resolveOptions.js\";\nimport type { Plugin, ViteDevServer } from \"vite\";\n\n/**\n * Dev server plugin for server environment.\n * Returns two plugins: one for HMR handling (all environments) and one for server config.\n */\nexport const vitePluginReactDevServer = function _vitePluginReactServerDevServer(options: StreamPluginOptions): Plugin[] {\n  if (options == null) {\n    throw new Error(\"options is required\");\n  }\n\n  const resolvedOptions = resolveOptions(options);\n  if (resolvedOptions.type === \"error\") {\n    if (resolvedOptions.error != null) {\n      throw resolvedOptions.error;\n    }\n    throw new Error(\"Failed to resolve options\");\n  }\n  const userOptions = resolvedOptions.userOptions;\n\n  // Separate plugin for HMR handling (must apply to all environments)\n  const hmrPlugin = {\n    name: \"vite-plugin-react-server:server-hmr\",\n    apply: \"serve\" as const,\n    // Server-level handleHotUpdate — sends custom WS event to client\n    handleHotUpdate({ file, server }: { file: string; server: ViteDevServer }) {\n      const moduleBase = userOptions.moduleBase || \"src\";\n      const projectRoot = userOptions.projectRoot || server.config.root;\n      const normalizedFile = file.replace(projectRoot, '').replace(/^\\/+/, '');\n      const isServerFile = normalizedFile.startsWith(moduleBase + '/') && \n        (file.endsWith('.tsx') || file.endsWith('.ts') || file.endsWith('.jsx') || file.endsWith('.js'));\n      \n      if (isServerFile) {\n        server.config.logger.info(`[vite-plugin-react-server] Server component changed: ${normalizedFile}`);\n        \n        // Send custom HMR event so client can refetch RSC stream\n        server.ws.send({\n          type: 'custom',\n          event: 'vite-plugin-react-server:server-component-update',\n          data: {\n            file: normalizedFile,\n            path: file,\n          },\n        });\n        \n        // Invalidate the server module so next request gets fresh content\n        const mod = server.environments['server']?.moduleGraph?.getModulesByFile(file);\n        if (mod) {\n          for (const m of mod) {\n            server.environments['server']?.moduleGraph?.invalidateModule(m);\n          }\n        }\n        \n        // Return empty array to prevent Vite's default full-page reload\n        // The client will refetch the RSC stream via the custom event\n        return [];\n      }\n    },\n  };\n\n  const serverPlugin = {\n    name: \"vite-plugin-react-server:dev-server-server\",\n    apply: \"serve\" as const,\n    applyToEnvironment(partialEnvironment: any) {\n      return partialEnvironment?.consumer === 'server';\n    },\n    configureServer(server: ViteDevServer) {\n      // Log that plugin is being configured\n      server.config.logger.info(`[vite-plugin-react-server] Dev server plugin configured for server environment (react-server condition)`);\n      \n      // Configure the React server for server environment (direct RSC processing)\n      // This uses the existing configureReactServer.server.js implementation\n      configureReactServer({\n        server,\n        autoDiscoveredFiles: {\n          propsMap: new Map(),\n          pageMap: new Map(),\n          rootMap: new Map(),\n          htmlMap: new Map(),\n          routeMap: new Map(),\n          urlMap: new Map(),\n          errors: [],\n          workerPaths: {},\n          serverEntry: null,\n          clientEntry: {},\n          clientInputs: {},\n          staticInputs: {},\n          serverInputs: {},\n          // staticManifest removed from AutoDiscoveredFiles\n          serverActions: {},\n        },\n        userOptions,\n        serverManifest: {},\n        resolvedConfig: server.config,\n      });\n    },\n  };\n\n  // Fix CJS named imports in server environment.\n  // Vite's esbuild JSX transform (and @vitejs/plugin-react if present) generates\n  // named imports like `import { useEffect } from \"react\"`. In the server environment,\n  // react's react-server export is CJS-only, and Node's ESM interop doesn't support\n  // named exports from CJS. This post-transform rewrites them.\n  const cjsFixPlugin: Plugin = {\n    name: \"vite-plugin-react-server:server-cjs-fix\",\n    apply: \"serve\" as const,\n    enforce: \"post\" as const,\n    applyToEnvironment(env: any) {\n      return env?.name === 'server';\n    },\n    transform(code: string, id: string) {\n      if (id.includes('node_modules')) return;\n      if (!id.match(/\\.[jt]sx?$/)) return;\n      \n      const namedImportRe = /import\\s*\\{([^}]+)\\}\\s*from\\s*[\"']react[\"']\\s*;?/g;\n      if (!namedImportRe.test(code)) return;\n      \n      namedImportRe.lastIndex = 0;\n      let counter = 0;\n      const result = code.replace(namedImportRe, (_match, imports) => {\n        const alias = `__react_cjs_${counter++}`;\n        return `import ${alias} from \"react\"; const {${imports}} = ${alias};`;\n      });\n      \n      return { code: result, map: null };\n    },\n  };\n\n  return [hmrPlugin, cjsFixPlugin, serverPlugin];\n};\n"],"names":[],"mappings":"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"}
|
|
158
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"plugin.server.js","sources":["../../../plugin/dev-server/plugin.server.ts"],"sourcesContent":["import type { StreamPluginOptions } from \"../../types.js\";\nimport { configureReactServer } from \"./configureReactServer.server.js\";\nimport { resolveOptions } from \"../config/resolveOptions.js\";\nimport type { Plugin, ViteDevServer } from \"vite\";\nimport { readFileSync } from \"node:fs\";\n\n/**\n * Dev server plugin for server environment.\n * Returns two plugins: one for HMR handling (all environments) and one for server config.\n */\nexport const vitePluginReactDevServer = function _vitePluginReactServerDevServer(options: StreamPluginOptions): Plugin[] {\n  if (options == null) {\n    throw new Error(\"options is required\");\n  }\n\n  const resolvedOptions = resolveOptions(options);\n  if (resolvedOptions.type === \"error\") {\n    if (resolvedOptions.error != null) {\n      throw resolvedOptions.error;\n    }\n    throw new Error(\"Failed to resolve options\");\n  }\n  const userOptions = resolvedOptions.userOptions;\n\n  // Separate plugin for HMR handling (must apply to all environments)\n  const hmrPlugin = {\n    name: \"vite-plugin-react-server:server-hmr\",\n    apply: \"serve\" as const,\n    // Server-level handleHotUpdate — sends custom WS event to client\n    // Vite 6 Environment API: hotUpdate runs per-environment.\n    // Prevent server/ssr environments from triggering page reload for client components.\n    hotUpdate(ctx: any) {\n      const { file, server } = ctx;\n      const envName = ctx.environment?.name ?? 'unknown';\n      \n      const moduleBase = userOptions.moduleBase || \"src\";\n      const projectRoot = userOptions.projectRoot || server?.config?.root || '';\n      const normalizedFile = file.replace(projectRoot, '').replace(/^\\/+/, '');\n      const isSourceFile = normalizedFile.startsWith(moduleBase + '/');\n      \n      if (!isSourceFile) return;\n      \n      // Client environment: let Vite/@vitejs/plugin-react handle it\n      if (envName === 'client') {\n        // Check if it's a client component — let Fast Refresh handle it\n        const isClient = (file.endsWith('.tsx') || file.endsWith('.ts') || file.endsWith('.jsx') || file.endsWith('.js')) && (() => {\n          try {\n            const head = readFileSync(file, 'utf-8').slice(0, 200);\n            return /^\\s*[\"']use client[\"']/.test(head.split('\\n')[0]);\n          } catch { return false; }\n        })();\n        \n        if (isClient) return; // Let Fast Refresh handle client components\n        \n        // Server component changed — send RSC refetch event to client\n        // Only do this once (from client env) to avoid duplicate events\n        server.config.logger.info(`[vite-plugin-react-server] File changed (RSC refetch): ${normalizedFile}`);\n        server.ws.send({\n          type: 'custom',\n          event: 'vite-plugin-react-server:server-component-update',\n          data: { file: normalizedFile, path: file },\n        });\n        \n        return []; // Don't trigger client-side page reload\n      }\n      \n      // Server/SSR environments: suppress page reload for all source files\n      // Server components are handled by the RSC refetch event sent above\n      // Invalidate the server module so next RSC request gets fresh content\n      if (envName === 'server') {\n        const mod = ctx.environment?.moduleGraph?.getModulesByFile(file);\n        if (mod) {\n          for (const m of mod) {\n            ctx.environment.moduleGraph.invalidateModule(m);\n          }\n        }\n      }\n      return [];\n    },\n    handleHotUpdate({ file, server }: { file: string; server: ViteDevServer }) {\n      const moduleBase = userOptions.moduleBase || \"src\";\n      const projectRoot = userOptions.projectRoot || server.config.root;\n      const normalizedFile = file.replace(projectRoot, '').replace(/^\\/+/, '');\n      const isSourceFile = normalizedFile.startsWith(moduleBase + '/') && \n        (file.endsWith('.tsx') || file.endsWith('.ts') || file.endsWith('.jsx') || file.endsWith('.js'));\n      \n      // Skip client components — let @vitejs/plugin-react handle them\n      // with Fast Refresh (preserves component-level state).\n      const isClientFile = isSourceFile && (() => {\n        try {\n          const head = readFileSync(file, 'utf-8').slice(0, 200);\n          return /^\\s*[\"']use client[\"']/.test(head.split('\\n')[0]);\n        } catch { return false; }\n      })();\n      \n      if (isSourceFile && !isClientFile) {\n        server.config.logger.info(`[vite-plugin-react-server] File changed (RSC refetch): ${normalizedFile}`);\n        \n        // Send custom HMR event so client can refetch RSC stream\n        server.ws.send({\n          type: 'custom',\n          event: 'vite-plugin-react-server:server-component-update',\n          data: {\n            file: normalizedFile,\n            path: file,\n          },\n        });\n        \n        // Invalidate the server module so next request gets fresh content\n        const mod = server.environments['server']?.moduleGraph?.getModulesByFile(file);\n        if (mod) {\n          for (const m of mod) {\n            server.environments['server']?.moduleGraph?.invalidateModule(m);\n          }\n        }\n        \n        // Return empty array to prevent Vite's default full-page reload\n        // The client will refetch the RSC stream via the custom event\n        return [];\n      }\n      \n      if (isClientFile) {\n        // Client components are handled by @vitejs/plugin-react (Fast Refresh)\n        // or Vite's client-side HMR. Return empty to prevent the server\n        // environment from triggering a full page reload.\n        return [];\n      }\n    },\n  };\n\n  const serverPlugin = {\n    name: \"vite-plugin-react-server:dev-server-server\",\n    apply: \"serve\" as const,\napplyToEnvironment(partialEnvironment: any) {\n      return partialEnvironment?.consumer === 'server';\n    },\n    configureServer(server: ViteDevServer) {\n      // Log that plugin is being configured\n      server.config.logger.info(`[vite-plugin-react-server] Dev server plugin configured for server environment (react-server condition)`);\n      \n      // Configure the React server for server environment (direct RSC processing)\n      // This uses the existing configureReactServer.server.js implementation\n      configureReactServer({\n        server,\n        autoDiscoveredFiles: {\n          propsMap: new Map(),\n          pageMap: new Map(),\n          rootMap: new Map(),\n          htmlMap: new Map(),\n          routeMap: new Map(),\n          urlMap: new Map(),\n          errors: [],\n          workerPaths: {},\n          serverEntry: null,\n          clientEntry: {},\n          clientInputs: {},\n          staticInputs: {},\n          serverInputs: {},\n          // staticManifest removed from AutoDiscoveredFiles\n          serverActions: {},\n        },\n        userOptions,\n        serverManifest: {},\n        resolvedConfig: server.config,\n      });\n    },\n  };\n\n  // Fix CJS named imports in server environment.\n  // Vite's esbuild JSX transform (and @vitejs/plugin-react if present) generates\n  // named imports like `import { useEffect } from \"react\"`. In the server environment,\n  // react's react-server export is CJS-only, and Node's ESM interop doesn't support\n  // named exports from CJS. This post-transform rewrites them.\n  const cjsFixPlugin: Plugin = {\n    name: \"vite-plugin-react-server:server-cjs-fix\",\n    apply: \"serve\" as const,\n    enforce: \"post\" as const,\n    applyToEnvironment(env: any) {\n      return env?.name === 'server';\n    },\n    transform(code: string, id: string) {\n      if (id.includes('node_modules')) return;\n      if (!id.match(/\\.[jt]sx?$/)) return;\n      \n      const namedImportRe = /import\\s*\\{([^}]+)\\}\\s*from\\s*[\"']react[\"']\\s*;?/g;\n      if (!namedImportRe.test(code)) return;\n      \n      namedImportRe.lastIndex = 0;\n      let counter = 0;\n      const result = code.replace(namedImportRe, (_match, imports) => {\n        const alias = `__react_cjs_${counter++}`;\n        return `import ${alias} from \"react\"; const {${imports}} = ${alias};`;\n      });\n      \n      return { code: result, map: null };\n    },\n  };\n\n  return [hmrPlugin, cjsFixPlugin, serverPlugin];\n};\n"],"names":[],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"plugin.server.d.ts","sourceRoot":"","sources":["../../../plugin/react-static/plugin.server.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAaH,OAAO,KAAK,EAIV,YAAY,EACb,MAAM,aAAa,CAAC;
|
|
1
|
+
{"version":3,"file":"plugin.server.d.ts","sourceRoot":"","sources":["../../../plugin/react-static/plugin.server.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAaH,OAAO,KAAK,EAIV,YAAY,EACb,MAAM,aAAa,CAAC;AA0CrB,eAAO,MAAM,iBAAiB,EAAE,YAwoB/B,CAAC"}
|
|
@@ -16,7 +16,8 @@
|
|
|
16
16
|
import { createLogger, } from "vite";
|
|
17
17
|
import { resolveOptions } from "../config/resolveOptions.js";
|
|
18
18
|
import { createBuildLoader } from "./createBuildLoader.server.js";
|
|
19
|
-
import {
|
|
19
|
+
import { renderPagesBatched } from "./renderPagesBatched.js";
|
|
20
|
+
import { renderPages as renderPagesSequential } from "./renderPages.js";
|
|
20
21
|
import { getBundleManifest } from "../helpers/getBundleManifest.js";
|
|
21
22
|
import { createWorker } from "../worker/createWorker.js";
|
|
22
23
|
import { serializedOptions, serializeResolvedConfig, } from "../helpers/serializeUserOptions.js";
|
|
@@ -375,6 +376,12 @@ export const reactStaticPlugin = function _reactStaticPlugin(options) {
|
|
|
375
376
|
}
|
|
376
377
|
}
|
|
377
378
|
}
|
|
379
|
+
// Select render mode based on build config
|
|
380
|
+
const renderMode = userOptions.build?.renderMode ?? "parallel";
|
|
381
|
+
const renderPages = renderMode === "sequential" ? renderPagesSequential : renderPagesBatched;
|
|
382
|
+
if (userOptions.verbose) {
|
|
383
|
+
logger.info(`[static] Using ${renderMode} rendering${renderMode === "parallel" ? ` (batch size: ${userOptions.build?.batchSize ?? 8})` : ""}`);
|
|
384
|
+
}
|
|
378
385
|
// this will render the routes
|
|
379
386
|
const renderPagesGenerator = renderPages(routes, {
|
|
380
387
|
...handlerOptions,
|
|
@@ -393,6 +400,7 @@ export const reactStaticPlugin = function _reactStaticPlugin(options) {
|
|
|
393
400
|
staticManifest: staticManifest, // Pass static manifest for path resolution
|
|
394
401
|
autoDiscoveredFiles: autoDiscoveredFiles,
|
|
395
402
|
cssFilesByPage: cssFilesByPage,
|
|
403
|
+
batchSize: userOptions.build?.batchSize,
|
|
396
404
|
}, renderPage);
|
|
397
405
|
// Process render results
|
|
398
406
|
let finalResult;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"renderPagesBatched.d.ts","sourceRoot":"","sources":["../../../plugin/react-static/renderPagesBatched.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAAE,aAAa,EAA2C,MAAM,YAAY,CAAC;
|
|
1
|
+
{"version":3,"file":"renderPagesBatched.d.ts","sourceRoot":"","sources":["../../../plugin/react-static/renderPagesBatched.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAAE,aAAa,EAA2C,MAAM,YAAY,CAAC;AA4OzF;;GAEG;AACH,eAAO,MAAM,kBAAkB,EAAE,aA6HhC,CAAC"}
|
|
@@ -1,5 +1,7 @@
|
|
|
1
1
|
import { handleError } from "../error/handleError.js";
|
|
2
2
|
import { fileWriter } from "./fileWriter.js";
|
|
3
|
+
import { createRenderMetrics } from "../metrics/createRenderMetrics.js";
|
|
4
|
+
import { createStreamMetrics } from "../metrics/createStreamMetrics.js";
|
|
3
5
|
const DEFAULT_BATCH_SIZE = 8;
|
|
4
6
|
function resolvePathWithManifest(path, manifest) {
|
|
5
7
|
const entry = manifest[path];
|
|
@@ -10,9 +12,10 @@ function resolvePathWithManifest(path, manifest) {
|
|
|
10
12
|
}
|
|
11
13
|
/**
|
|
12
14
|
* Renders a single route completely, consuming all yields from renderPage
|
|
13
|
-
* and writing the RSC and HTML files
|
|
15
|
+
* and writing the RSC and HTML files. Collects metrics and handles events
|
|
16
|
+
* identically to the sequential renderPages.
|
|
14
17
|
*/
|
|
15
|
-
async function renderSingleRoute(route, handlerOptions, renderPage, manifest) {
|
|
18
|
+
async function renderSingleRoute(route, handlerOptions, renderPage, manifest, failedRoutes) {
|
|
16
19
|
const { autoDiscoveredFiles, cssFilesByPage, ...options } = handlerOptions;
|
|
17
20
|
const { page, props, root, html } = autoDiscoveredFiles.urlMap?.get(route) || {};
|
|
18
21
|
if (!page) {
|
|
@@ -23,6 +26,120 @@ async function renderSingleRoute(route, handlerOptions, renderPage, manifest) {
|
|
|
23
26
|
const resolvedPropsPath = props ? resolvePathWithManifest(props, manifest) : undefined;
|
|
24
27
|
const resolvedRootPath = root ? resolvePathWithManifest(root, manifest) : undefined;
|
|
25
28
|
const resolvedHtmlPath = html ? resolvePathWithManifest(html, manifest) : undefined;
|
|
29
|
+
// Store results for metrics tracking
|
|
30
|
+
const routeResults = new Map();
|
|
31
|
+
// Create onEvent wrapper that handles route.error and metrics collection
|
|
32
|
+
// This mirrors the sequential renderPages behavior exactly
|
|
33
|
+
const wrapperOnEvent = (event) => {
|
|
34
|
+
// Call the original onEvent first
|
|
35
|
+
if (options.onEvent) {
|
|
36
|
+
options.onEvent(event);
|
|
37
|
+
}
|
|
38
|
+
// Handle route.error events
|
|
39
|
+
if (event.type === "route.error") {
|
|
40
|
+
const detectedPanicError = handleError({
|
|
41
|
+
error: event.data.error,
|
|
42
|
+
logger: options.logger,
|
|
43
|
+
panicThreshold: event.data.panicThreshold,
|
|
44
|
+
context: `route.error (${event.data.route})`,
|
|
45
|
+
});
|
|
46
|
+
if (detectedPanicError != null) {
|
|
47
|
+
options.logger?.error(`[renderPagesBatched] Panic error for route ${event.data.route}: ${event.data.error.message}`);
|
|
48
|
+
failedRoutes.set(event.data.route, event.data.error);
|
|
49
|
+
}
|
|
50
|
+
else {
|
|
51
|
+
options.logger?.warn(`[renderPagesBatched] Non-panic error for route ${event.data.route}: ${event.data.error.message}`);
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
// Handle metrics collection on file.write.done
|
|
55
|
+
if (event.type === "file.write.done" && event.data.route === route) {
|
|
56
|
+
const routeResult = routeResults.get(route);
|
|
57
|
+
if (routeResult && routeResult.type === "success") {
|
|
58
|
+
if (event.data.fileType === "html") {
|
|
59
|
+
const endTime = performance.now();
|
|
60
|
+
const htmlMetrics = createRenderMetrics({
|
|
61
|
+
route: route,
|
|
62
|
+
type: routeResult.metrics.html.type,
|
|
63
|
+
fromMainThread: routeResult.metrics.html.fromMainThread,
|
|
64
|
+
fromRscWorker: routeResult.metrics.html.fromRscWorker,
|
|
65
|
+
fromHtmlWorker: routeResult.metrics.html.fromHtmlWorker,
|
|
66
|
+
fileSize: event.data.content.length,
|
|
67
|
+
chunks: event.data.chunks || 0,
|
|
68
|
+
processingTime: endTime - routeResult.metrics.html.streamMetrics.startTime,
|
|
69
|
+
chunkRate: (event.data.chunks || 0) / ((endTime - routeResult.metrics.html.streamMetrics.startTime) / 1000),
|
|
70
|
+
fileName: event.data.fileName,
|
|
71
|
+
outputPath: event.data.path,
|
|
72
|
+
baseDir: event.data.baseDir,
|
|
73
|
+
routePath: event.data.routePath,
|
|
74
|
+
streamMetrics: createStreamMetrics({
|
|
75
|
+
...routeResult.metrics.html.streamMetrics,
|
|
76
|
+
chunks: event.data.chunks || 0,
|
|
77
|
+
bytes: event.data.content.length,
|
|
78
|
+
duration: endTime - routeResult.metrics.html.streamMetrics.startTime,
|
|
79
|
+
endTime: endTime,
|
|
80
|
+
}),
|
|
81
|
+
});
|
|
82
|
+
if (options.onMetrics) {
|
|
83
|
+
options.onMetrics(htmlMetrics);
|
|
84
|
+
}
|
|
85
|
+
// Also emit RSC Full metrics if available
|
|
86
|
+
if (routeResult.metrics?.rscFull) {
|
|
87
|
+
const rscFullEndTime = performance.now();
|
|
88
|
+
const rscFullMetrics = createRenderMetrics({
|
|
89
|
+
route: route,
|
|
90
|
+
type: routeResult.metrics.rscFull.type,
|
|
91
|
+
fromMainThread: routeResult.metrics.rscFull.fromMainThread,
|
|
92
|
+
fromRscWorker: routeResult.metrics.rscFull.fromRscWorker,
|
|
93
|
+
fromHtmlWorker: routeResult.metrics.rscFull.fromHtmlWorker,
|
|
94
|
+
processingTime: rscFullEndTime - routeResult.metrics.rscFull.streamMetrics.startTime,
|
|
95
|
+
chunks: routeResult.metrics.rscFull.streamMetrics.chunks,
|
|
96
|
+
chunkRate: routeResult.metrics.rscFull.streamMetrics.chunks / ((rscFullEndTime - routeResult.metrics.rscFull.streamMetrics.startTime) / 1000),
|
|
97
|
+
fileName: event.data.fileName,
|
|
98
|
+
outputPath: event.data.path,
|
|
99
|
+
baseDir: event.data.baseDir,
|
|
100
|
+
routePath: event.data.routePath,
|
|
101
|
+
streamMetrics: createStreamMetrics({
|
|
102
|
+
...routeResult.metrics.rscFull.streamMetrics,
|
|
103
|
+
duration: rscFullEndTime - routeResult.metrics.rscFull.streamMetrics.startTime,
|
|
104
|
+
endTime: rscFullEndTime,
|
|
105
|
+
}),
|
|
106
|
+
});
|
|
107
|
+
if (options.onMetrics) {
|
|
108
|
+
options.onMetrics(rscFullMetrics);
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
else if (event.data.fileType === "rsc") {
|
|
113
|
+
const rscEndTime = performance.now();
|
|
114
|
+
const rscMetrics = createRenderMetrics({
|
|
115
|
+
route: route,
|
|
116
|
+
type: routeResult.metrics.rscHeadless.type,
|
|
117
|
+
fromMainThread: routeResult.metrics.rscHeadless.fromMainThread,
|
|
118
|
+
fromRscWorker: routeResult.metrics.rscHeadless.fromRscWorker,
|
|
119
|
+
fromHtmlWorker: routeResult.metrics.rscHeadless.fromHtmlWorker,
|
|
120
|
+
fileSize: event.data.content.length,
|
|
121
|
+
chunks: event.data.chunks || 0,
|
|
122
|
+
processingTime: rscEndTime - routeResult.metrics.rscHeadless.streamMetrics.startTime,
|
|
123
|
+
chunkRate: (event.data.chunks || 0) / ((rscEndTime - routeResult.metrics.rscHeadless.streamMetrics.startTime) / 1000),
|
|
124
|
+
fileName: event.data.fileName,
|
|
125
|
+
outputPath: event.data.path,
|
|
126
|
+
baseDir: event.data.baseDir,
|
|
127
|
+
routePath: event.data.routePath,
|
|
128
|
+
streamMetrics: createStreamMetrics({
|
|
129
|
+
...routeResult.metrics.rscHeadless.streamMetrics,
|
|
130
|
+
chunks: event.data.chunks || 0,
|
|
131
|
+
bytes: event.data.content.length,
|
|
132
|
+
duration: rscEndTime - routeResult.metrics.rscHeadless.streamMetrics.startTime,
|
|
133
|
+
endTime: rscEndTime,
|
|
134
|
+
}),
|
|
135
|
+
});
|
|
136
|
+
if (options.onMetrics) {
|
|
137
|
+
options.onMetrics(rscMetrics);
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
};
|
|
26
143
|
const routeHandlerOptions = {
|
|
27
144
|
...options,
|
|
28
145
|
manifest,
|
|
@@ -34,6 +151,7 @@ async function renderSingleRoute(route, handlerOptions, renderPage, manifest) {
|
|
|
34
151
|
cssFiles: cssFilesByPage?.get(route) ?? new Map(),
|
|
35
152
|
globalCss: options.globalCss ?? new Map(),
|
|
36
153
|
id: `${route}-${Date.now()}-${Math.random().toString(36).substring(2, 11)}`,
|
|
154
|
+
onEvent: wrapperOnEvent,
|
|
37
155
|
};
|
|
38
156
|
const pageRenderer = renderPage(routeHandlerOptions);
|
|
39
157
|
const results = [];
|
|
@@ -41,18 +159,17 @@ async function renderSingleRoute(route, handlerOptions, renderPage, manifest) {
|
|
|
41
159
|
// Consume all yields from the page renderer and write files
|
|
42
160
|
for await (const result of pageRenderer) {
|
|
43
161
|
results.push(result);
|
|
44
|
-
// Track error results
|
|
45
162
|
if (result.type === "error" && result.error) {
|
|
46
163
|
routeError = result.error instanceof Error ? result.error : new Error(String(result.error));
|
|
47
164
|
}
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
const
|
|
165
|
+
if (result.type === "success" || result.type === "skip") {
|
|
166
|
+
// Store result for metrics tracking (wrapperOnEvent needs this)
|
|
167
|
+
routeResults.set(route, result);
|
|
168
|
+
const rscWritePromise = fileWriter(result.rsc, "rsc", { ...options, route, onEvent: wrapperOnEvent, logger: options.logger }, options.signal);
|
|
169
|
+
const htmlWritePromise = fileWriter(result.html, "html", { ...options, route, onEvent: wrapperOnEvent, logger: options.logger }, options.signal);
|
|
52
170
|
await Promise.all([rscWritePromise, htmlWritePromise]);
|
|
53
171
|
}
|
|
54
172
|
}
|
|
55
|
-
// Return error if any result was an error
|
|
56
173
|
if (routeError) {
|
|
57
174
|
return { route, results, error: routeError };
|
|
58
175
|
}
|
|
@@ -116,7 +233,7 @@ export const renderPagesBatched = (routes, handlerOptions, renderPage) => {
|
|
|
116
233
|
return abortResult;
|
|
117
234
|
}
|
|
118
235
|
// Render all pages in this batch concurrently
|
|
119
|
-
const batchPromises = batch.map(route => renderSingleRoute(route, handlerOptions, renderPage, manifest));
|
|
236
|
+
const batchPromises = batch.map(route => renderSingleRoute(route, handlerOptions, renderPage, manifest, failedRoutes));
|
|
120
237
|
const batchResults = await Promise.all(batchPromises);
|
|
121
238
|
// Process results from this batch
|
|
122
239
|
for (const { route, results: pageResults, error } of batchResults) {
|
|
@@ -130,6 +247,16 @@ export const renderPagesBatched = (routes, handlerOptions, renderPage) => {
|
|
|
130
247
|
if (panicError != null) {
|
|
131
248
|
failedRoutes.set(route, error);
|
|
132
249
|
options.logger?.error(`[renderPagesBatched] Panic error for route ${route}: ${error.message}`);
|
|
250
|
+
const errorResult = {
|
|
251
|
+
type: "error",
|
|
252
|
+
error,
|
|
253
|
+
route,
|
|
254
|
+
failedRoutes,
|
|
255
|
+
completedRoutes,
|
|
256
|
+
results,
|
|
257
|
+
};
|
|
258
|
+
yield errorResult;
|
|
259
|
+
return errorResult;
|
|
133
260
|
}
|
|
134
261
|
else {
|
|
135
262
|
options.logger?.warn(`[renderPagesBatched] Non-panic error for route ${route}: ${error.message}`);
|
|
@@ -137,7 +264,6 @@ export const renderPagesBatched = (routes, handlerOptions, renderPage) => {
|
|
|
137
264
|
}
|
|
138
265
|
else {
|
|
139
266
|
completedRoutes.add(route);
|
|
140
|
-
// Yield each result from this page
|
|
141
267
|
for (const result of pageResults) {
|
|
142
268
|
if (result.type === "success" || result.type === "skip") {
|
|
143
269
|
results.set(route, result);
|
|
@@ -156,32 +282,6 @@ export const renderPagesBatched = (routes, handlerOptions, renderPage) => {
|
|
|
156
282
|
options.logger?.info(`[renderPagesBatched] Completed batch: ${completedRoutes.size}/${routeArray.length} pages`);
|
|
157
283
|
}
|
|
158
284
|
}
|
|
159
|
-
// Check if we should panic based on failed routes
|
|
160
|
-
if (failedRoutes.size > 0) {
|
|
161
|
-
const firstError = Array.from(failedRoutes.values())[0];
|
|
162
|
-
const panicError = handleError({
|
|
163
|
-
error: firstError,
|
|
164
|
-
logger: options.logger,
|
|
165
|
-
panicThreshold: options.panicThreshold,
|
|
166
|
-
context: `renderPagesBatched final check`,
|
|
167
|
-
});
|
|
168
|
-
if (panicError != null) {
|
|
169
|
-
if (options.verbose) {
|
|
170
|
-
options.logger?.error(`[renderPagesBatched] Build failed due to panic threshold: ${failedRoutes.size} routes failed`);
|
|
171
|
-
}
|
|
172
|
-
// Yield error before returning
|
|
173
|
-
const errorResult = {
|
|
174
|
-
type: "error",
|
|
175
|
-
error: panicError,
|
|
176
|
-
route: "",
|
|
177
|
-
failedRoutes,
|
|
178
|
-
completedRoutes,
|
|
179
|
-
results,
|
|
180
|
-
};
|
|
181
|
-
yield errorResult;
|
|
182
|
-
return errorResult;
|
|
183
|
-
}
|
|
184
|
-
}
|
|
185
285
|
// Final success result
|
|
186
286
|
const finalResult = {
|
|
187
287
|
type: "success",
|
|
@@ -59,6 +59,7 @@ export type RenderPagesHandlerOptions = Omit<CreateHandlerOptions, "pagePath" |
|
|
|
59
59
|
cssFilesByPage: Map<string, Map<string, CssContent>>;
|
|
60
60
|
serverPipeableStreamOptions: any;
|
|
61
61
|
staticManifest?: Manifest;
|
|
62
|
+
batchSize?: number;
|
|
62
63
|
};
|
|
63
64
|
export type RenderPagesFn = (routes: string[], handlerOptions: RenderPagesHandlerOptions, renderPage: RenderPageFn) => RenderPagesReturn;
|
|
64
65
|
export type RenderPageReturn = AsyncGenerator<RenderPageResult, void, unknown>;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../../plugin/react-static/types.ts"],"names":[],"mappings":"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
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../../plugin/react-static/types.ts"],"names":[],"mappings":"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"}
|
package/dist/plugin/types.d.ts
CHANGED
|
@@ -724,6 +724,21 @@ export type BuildConfig = {
|
|
|
724
724
|
rscExtension?: string;
|
|
725
725
|
cssModuleExtension?: string;
|
|
726
726
|
nodeExtension?: string;
|
|
727
|
+
/**
|
|
728
|
+
* Controls how pages are rendered during static generation.
|
|
729
|
+
*
|
|
730
|
+
* - `"parallel"` (default): Renders pages in concurrent batches for faster builds.
|
|
731
|
+
* Use `batchSize` to control concurrency (default: 8).
|
|
732
|
+
* - `"sequential"`: Renders pages one at a time. Slower but uses less memory
|
|
733
|
+
* and produces deterministic output order.
|
|
734
|
+
*/
|
|
735
|
+
renderMode?: "parallel" | "sequential";
|
|
736
|
+
/**
|
|
737
|
+
* Number of pages to render concurrently when `renderMode` is `"parallel"`.
|
|
738
|
+
* Higher values use more memory but build faster.
|
|
739
|
+
* @default 8
|
|
740
|
+
*/
|
|
741
|
+
batchSize?: number;
|
|
727
742
|
};
|
|
728
743
|
export type DevConfig = {
|
|
729
744
|
useHtmlWorker?: boolean | undefined;
|