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.
Files changed (48) hide show
  1. package/README.md +32 -18
  2. package/dist/package.json +4 -2
  3. package/dist/plugin/config/resolveOptions.d.ts.map +1 -1
  4. package/dist/plugin/config/resolveOptions.js +4 -2
  5. package/dist/plugin/dev-server/plugin.client.d.ts.map +1 -1
  6. package/dist/plugin/dev-server/plugin.client.js +13 -3
  7. package/dist/plugin/dev-server/plugin.server.d.ts.map +1 -1
  8. package/dist/plugin/dev-server/plugin.server.js +54 -4
  9. package/dist/plugin/react-static/plugin.server.d.ts.map +1 -1
  10. package/dist/plugin/react-static/plugin.server.js +9 -1
  11. package/dist/plugin/react-static/renderPagesBatched.d.ts.map +1 -1
  12. package/dist/plugin/react-static/renderPagesBatched.js +136 -36
  13. package/dist/plugin/react-static/types.d.ts +1 -0
  14. package/dist/plugin/react-static/types.d.ts.map +1 -1
  15. package/dist/plugin/types.d.ts +15 -0
  16. package/dist/plugin/types.d.ts.map +1 -1
  17. package/dist/plugin/utils/createReactFetcher.js +24 -2
  18. package/dist/plugin/utils/useRscHmr.js +10 -2
  19. package/dist/plugin/vendor/register-vendor.js +1 -1
  20. package/dist/plugin/vendor/vendor-alias.d.ts +5 -2
  21. package/dist/plugin/vendor/vendor-alias.d.ts.map +1 -1
  22. package/dist/plugin/vendor/vendor-alias.js +60 -26
  23. package/dist/plugin/vendor/vendor.server.d.ts.map +1 -1
  24. package/dist/plugin/vendor/vendor.server.js +2 -2
  25. package/dist/plugin/vendor/vendor.static.d.ts.map +1 -1
  26. package/dist/plugin/vendor/vendor.static.js +2 -2
  27. package/dist/tsconfig.tsbuildinfo +1 -1
  28. package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-client.browser.development.js +1 -1
  29. package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-client.browser.production.js +1 -1
  30. package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-client.node.development.js +1 -1
  31. package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-client.node.production.js +1 -1
  32. package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-server.node.development.js +1 -1
  33. package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-server.node.production.js +1 -1
  34. package/oss-experimental/react-server-dom-esm/esm/react-server-dom-esm-client.browser.development.js +1 -1
  35. package/oss-experimental/react-server-dom-esm/esm/react-server-dom-esm-client.browser.production.js +1 -1
  36. package/oss-experimental/react-server-dom-esm/package.json +3 -3
  37. package/package.json +4 -2
  38. package/plugin/config/resolveOptions.ts +2 -0
  39. package/plugin/dev-server/plugin.client.ts +13 -2
  40. package/plugin/dev-server/plugin.server.ts +71 -4
  41. package/plugin/react-static/plugin.server.ts +11 -1
  42. package/plugin/react-static/renderPagesBatched.ts +148 -39
  43. package/plugin/react-static/types.ts +1 -0
  44. package/plugin/types.ts +15 -0
  45. package/plugin/vendor/register-vendor.ts +1 -1
  46. package/plugin/vendor/vendor-alias.ts +61 -39
  47. package/plugin/vendor/vendor.server.ts +3 -3
  48. 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 isServerFile = normalizedFile.startsWith(moduleBase + "/") && (file.endsWith(".tsx") || file.endsWith(".ts") || file.endsWith(".jsx") || file.endsWith(".js"));
30
- if (isServerFile) {
31
- server.config.logger.info(`[vite-plugin-react-server] Server component changed: ${normalizedFile}`);
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":";;;;;;;;AASa,MAAA,wBAAA,GAA2B,SAAS,+BAAA,CAAgC,OAAwC,EAAA;AACvH,EAAA,IAAI,WAAW,IAAM,EAAA;AACnB,IAAM,MAAA,IAAI,MAAM,qBAAqB,CAAA;AAAA;AAGvC,EAAM,MAAA,eAAA,GAAkB,eAAe,OAAO,CAAA;AAC9C,EAAI,IAAA,eAAA,CAAgB,SAAS,OAAS,EAAA;AACpC,IAAI,IAAA,eAAA,CAAgB,SAAS,IAAM,EAAA;AACjC,MAAA,MAAM,eAAgB,CAAA,KAAA;AAAA;AAExB,IAAM,MAAA,IAAI,MAAM,2BAA2B,CAAA;AAAA;AAE7C,EAAA,MAAM,cAAc,eAAgB,CAAA,WAAA;AAGpC,EAAA,MAAM,SAAY,GAAA;AAAA,IAChB,IAAM,EAAA,qCAAA;AAAA,IACN,KAAO,EAAA,OAAA;AAAA;AAAA,IAEP,eAAgB,CAAA,EAAE,IAAM,EAAA,MAAA,EAAmD,EAAA;AACzE,MAAM,MAAA,UAAA,GAAa,YAAY,UAAc,IAAA,KAAA;AAC7C,MAAA,MAAM,WAAc,GAAA,WAAA,CAAY,WAAe,IAAA,MAAA,CAAO,MAAO,CAAA,IAAA;AAC7D,MAAM,MAAA,cAAA,GAAiB,KAAK,OAAQ,CAAA,WAAA,EAAa,EAAE,CAAE,CAAA,OAAA,CAAQ,QAAQ,EAAE,CAAA;AACvE,MAAM,MAAA,YAAA,GAAe,eAAe,UAAW,CAAA,UAAA,GAAa,GAAG,CAC5D,KAAA,IAAA,CAAK,SAAS,MAAM,CAAA,IAAK,KAAK,QAAS,CAAA,KAAK,KAAK,IAAK,CAAA,QAAA,CAAS,MAAM,CAAK,IAAA,IAAA,CAAK,SAAS,KAAK,CAAA,CAAA;AAEhG,MAAA,IAAI,YAAc,EAAA;AAChB,QAAA,MAAA,CAAO,MAAO,CAAA,MAAA,CAAO,IAAK,CAAA,CAAA,qDAAA,EAAwD,cAAc,CAAE,CAAA,CAAA;AAGlG,QAAA,MAAA,CAAO,GAAG,IAAK,CAAA;AAAA,UACb,IAAM,EAAA,QAAA;AAAA,UACN,KAAO,EAAA,kDAAA;AAAA,UACP,IAAM,EAAA;AAAA,YACJ,IAAM,EAAA,cAAA;AAAA,YACN,IAAM,EAAA;AAAA;AACR,SACD,CAAA;AAGD,QAAA,MAAM,MAAM,MAAO,CAAA,YAAA,CAAa,QAAQ,CAAG,EAAA,WAAA,EAAa,iBAAiB,IAAI,CAAA;AAC7E,QAAA,IAAI,GAAK,EAAA;AACP,UAAA,KAAA,MAAW,KAAK,GAAK,EAAA;AACnB,YAAA,MAAA,CAAO,YAAa,CAAA,QAAQ,CAAG,EAAA,WAAA,EAAa,iBAAiB,CAAC,CAAA;AAAA;AAChE;AAKF,QAAA,OAAO,EAAC;AAAA;AACV;AACF,GACF;AAEA,EAAA,MAAM,YAAe,GAAA;AAAA,IACnB,IAAM,EAAA,4CAAA;AAAA,IACN,KAAO,EAAA,OAAA;AAAA,IACP,mBAAmB,kBAAyB,EAAA;AAC1C,MAAA,OAAO,oBAAoB,QAAa,KAAA,QAAA;AAAA,KAC1C;AAAA,IACA,gBAAgB,MAAuB,EAAA;AAErC,MAAO,MAAA,CAAA,MAAA,CAAO,MAAO,CAAA,IAAA,CAAK,CAAyG,uGAAA,CAAA,CAAA;AAInI,MAAqB,oBAAA,CAAA;AAAA,QACnB,MAAA;AAAA,QACA,mBAAqB,EAAA;AAAA,UACnB,QAAA,sBAAc,GAAI,EAAA;AAAA,UAClB,OAAA,sBAAa,GAAI,EAAA;AAAA,UACjB,OAAA,sBAAa,GAAI,EAAA;AAAA,UACjB,OAAA,sBAAa,GAAI,EAAA;AAAA,UACjB,QAAA,sBAAc,GAAI,EAAA;AAAA,UAClB,MAAA,sBAAY,GAAI,EAAA;AAAA,UAChB,QAAQ,EAAC;AAAA,UACT,aAAa,EAAC;AAAA,UACd,WAAa,EAAA,IAAA;AAAA,UACb,aAAa,EAAC;AAAA,UACd,cAAc,EAAC;AAAA,UACf,cAAc,EAAC;AAAA,UACf,cAAc,EAAC;AAAA;AAAA,UAEf,eAAe;AAAC,SAClB;AAAA,QACA,WAAA;AAAA,QACA,gBAAgB,EAAC;AAAA,QACjB,gBAAgB,MAAO,CAAA;AAAA,OACxB,CAAA;AAAA;AACH,GACF;AAOA,EAAA,MAAM,YAAuB,GAAA;AAAA,IAC3B,IAAM,EAAA,yCAAA;AAAA,IACN,KAAO,EAAA,OAAA;AAAA,IACP,OAAS,EAAA,MAAA;AAAA,IACT,mBAAmB,GAAU,EAAA;AAC3B,MAAA,OAAO,KAAK,IAAS,KAAA,QAAA;AAAA,KACvB;AAAA,IACA,SAAA,CAAU,MAAc,EAAY,EAAA;AAClC,MAAI,IAAA,EAAA,CAAG,QAAS,CAAA,cAAc,CAAG,EAAA;AACjC,MAAA,IAAI,CAAC,EAAA,CAAG,KAAM,CAAA,YAAY,CAAG,EAAA;AAE7B,MAAA,MAAM,aAAgB,GAAA,mDAAA;AACtB,MAAA,IAAI,CAAC,aAAA,CAAc,IAAK,CAAA,IAAI,CAAG,EAAA;AAE/B,MAAA,aAAA,CAAc,SAAY,GAAA,CAAA;AAC1B,MAAA,IAAI,OAAU,GAAA,CAAA;AACd,MAAA,MAAM,SAAS,IAAK,CAAA,OAAA,CAAQ,aAAe,EAAA,CAAC,QAAQ,OAAY,KAAA;AAC9D,QAAM,MAAA,KAAA,GAAQ,eAAe,OAAS,EAAA,CAAA,CAAA;AACtC,QAAA,OAAO,CAAU,OAAA,EAAA,KAAK,CAAyB,sBAAA,EAAA,OAAO,OAAO,KAAK,CAAA,CAAA,CAAA;AAAA,OACnE,CAAA;AAED,MAAA,OAAO,EAAE,IAAA,EAAM,MAAQ,EAAA,GAAA,EAAK,IAAK,EAAA;AAAA;AACnC,GACF;AAEA,EAAO,OAAA,CAAC,SAAW,EAAA,YAAA,EAAc,YAAY,CAAA;AAC/C;;;;"}
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":";;;;;;;;;AAUa,MAAA,wBAAA,GAA2B,SAAS,+BAAA,CAAgC,OAAwC,EAAA;AACvH,EAAA,IAAI,WAAW,IAAM,EAAA;AACnB,IAAM,MAAA,IAAI,MAAM,qBAAqB,CAAA;AAAA;AAGvC,EAAM,MAAA,eAAA,GAAkB,eAAe,OAAO,CAAA;AAC9C,EAAI,IAAA,eAAA,CAAgB,SAAS,OAAS,EAAA;AACpC,IAAI,IAAA,eAAA,CAAgB,SAAS,IAAM,EAAA;AACjC,MAAA,MAAM,eAAgB,CAAA,KAAA;AAAA;AAExB,IAAM,MAAA,IAAI,MAAM,2BAA2B,CAAA;AAAA;AAE7C,EAAA,MAAM,cAAc,eAAgB,CAAA,WAAA;AAGpC,EAAA,MAAM,SAAY,GAAA;AAAA,IAChB,IAAM,EAAA,qCAAA;AAAA,IACN,KAAO,EAAA,OAAA;AAAA;AAAA;AAAA;AAAA,IAIP,UAAU,GAAU,EAAA;AAClB,MAAM,MAAA,EAAE,IAAM,EAAA,MAAA,EAAW,GAAA,GAAA;AACzB,MAAM,MAAA,OAAA,GAAU,GAAI,CAAA,WAAA,EAAa,IAAQ,IAAA,SAAA;AAEzC,MAAM,MAAA,UAAA,GAAa,YAAY,UAAc,IAAA,KAAA;AAC7C,MAAA,MAAM,WAAc,GAAA,WAAA,CAAY,WAAe,IAAA,MAAA,EAAQ,QAAQ,IAAQ,IAAA,EAAA;AACvE,MAAM,MAAA,cAAA,GAAiB,KAAK,OAAQ,CAAA,WAAA,EAAa,EAAE,CAAE,CAAA,OAAA,CAAQ,QAAQ,EAAE,CAAA;AACvE,MAAA,MAAM,YAAe,GAAA,cAAA,CAAe,UAAW,CAAA,UAAA,GAAa,GAAG,CAAA;AAE/D,MAAA,IAAI,CAAC,YAAc,EAAA;AAGnB,MAAA,IAAI,YAAY,QAAU,EAAA;AAExB,QAAA,MAAM,YAAY,IAAK,CAAA,QAAA,CAAS,MAAM,CAAA,IAAK,KAAK,QAAS,CAAA,KAAK,CAAK,IAAA,IAAA,CAAK,SAAS,MAAM,CAAA,IAAK,KAAK,QAAS,CAAA,KAAK,OAAO,MAAM;AAC1H,UAAI,IAAA;AACF,YAAA,MAAM,OAAO,YAAa,CAAA,IAAA,EAAM,OAAO,CAAE,CAAA,KAAA,CAAM,GAAG,GAAG,CAAA;AACrD,YAAA,OAAO,yBAAyB,IAAK,CAAA,IAAA,CAAK,MAAM,IAAI,CAAA,CAAE,CAAC,CAAC,CAAA;AAAA,WAClD,CAAA,MAAA;AAAE,YAAO,OAAA,KAAA;AAAA;AAAO,SACvB,GAAA;AAEH,QAAA,IAAI,QAAU,EAAA;AAId,QAAA,MAAA,CAAO,MAAO,CAAA,MAAA,CAAO,IAAK,CAAA,CAAA,uDAAA,EAA0D,cAAc,CAAE,CAAA,CAAA;AACpG,QAAA,MAAA,CAAO,GAAG,IAAK,CAAA;AAAA,UACb,IAAM,EAAA,QAAA;AAAA,UACN,KAAO,EAAA,kDAAA;AAAA,UACP,IAAM,EAAA,EAAE,IAAM,EAAA,cAAA,EAAgB,MAAM,IAAK;AAAA,SAC1C,CAAA;AAED,QAAA,OAAO,EAAC;AAAA;AAMV,MAAA,IAAI,YAAY,QAAU,EAAA;AACxB,QAAA,MAAM,GAAM,GAAA,GAAA,CAAI,WAAa,EAAA,WAAA,EAAa,iBAAiB,IAAI,CAAA;AAC/D,QAAA,IAAI,GAAK,EAAA;AACP,UAAA,KAAA,MAAW,KAAK,GAAK,EAAA;AACnB,YAAI,GAAA,CAAA,WAAA,CAAY,WAAY,CAAA,gBAAA,CAAiB,CAAC,CAAA;AAAA;AAChD;AACF;AAEF,MAAA,OAAO,EAAC;AAAA,KACV;AAAA,IACA,eAAgB,CAAA,EAAE,IAAM,EAAA,MAAA,EAAmD,EAAA;AACzE,MAAM,MAAA,UAAA,GAAa,YAAY,UAAc,IAAA,KAAA;AAC7C,MAAA,MAAM,WAAc,GAAA,WAAA,CAAY,WAAe,IAAA,MAAA,CAAO,MAAO,CAAA,IAAA;AAC7D,MAAM,MAAA,cAAA,GAAiB,KAAK,OAAQ,CAAA,WAAA,EAAa,EAAE,CAAE,CAAA,OAAA,CAAQ,QAAQ,EAAE,CAAA;AACvE,MAAM,MAAA,YAAA,GAAe,eAAe,UAAW,CAAA,UAAA,GAAa,GAAG,CAC5D,KAAA,IAAA,CAAK,SAAS,MAAM,CAAA,IAAK,KAAK,QAAS,CAAA,KAAK,KAAK,IAAK,CAAA,QAAA,CAAS,MAAM,CAAK,IAAA,IAAA,CAAK,SAAS,KAAK,CAAA,CAAA;AAIhG,MAAM,MAAA,YAAA,GAAe,iBAAiB,MAAM;AAC1C,QAAI,IAAA;AACF,UAAA,MAAM,OAAO,YAAa,CAAA,IAAA,EAAM,OAAO,CAAE,CAAA,KAAA,CAAM,GAAG,GAAG,CAAA;AACrD,UAAA,OAAO,yBAAyB,IAAK,CAAA,IAAA,CAAK,MAAM,IAAI,CAAA,CAAE,CAAC,CAAC,CAAA;AAAA,SAClD,CAAA,MAAA;AAAE,UAAO,OAAA,KAAA;AAAA;AAAO,OACvB,GAAA;AAEH,MAAI,IAAA,YAAA,IAAgB,CAAC,YAAc,EAAA;AACjC,QAAA,MAAA,CAAO,MAAO,CAAA,MAAA,CAAO,IAAK,CAAA,CAAA,uDAAA,EAA0D,cAAc,CAAE,CAAA,CAAA;AAGpG,QAAA,MAAA,CAAO,GAAG,IAAK,CAAA;AAAA,UACb,IAAM,EAAA,QAAA;AAAA,UACN,KAAO,EAAA,kDAAA;AAAA,UACP,IAAM,EAAA;AAAA,YACJ,IAAM,EAAA,cAAA;AAAA,YACN,IAAM,EAAA;AAAA;AACR,SACD,CAAA;AAGD,QAAA,MAAM,MAAM,MAAO,CAAA,YAAA,CAAa,QAAQ,CAAG,EAAA,WAAA,EAAa,iBAAiB,IAAI,CAAA;AAC7E,QAAA,IAAI,GAAK,EAAA;AACP,UAAA,KAAA,MAAW,KAAK,GAAK,EAAA;AACnB,YAAA,MAAA,CAAO,YAAa,CAAA,QAAQ,CAAG,EAAA,WAAA,EAAa,iBAAiB,CAAC,CAAA;AAAA;AAChE;AAKF,QAAA,OAAO,EAAC;AAAA;AAGV,MAAA,IAAI,YAAc,EAAA;AAIhB,QAAA,OAAO,EAAC;AAAA;AACV;AACF,GACF;AAEA,EAAA,MAAM,YAAe,GAAA;AAAA,IACnB,IAAM,EAAA,4CAAA;AAAA,IACN,KAAO,EAAA,OAAA;AAAA,IACX,mBAAmB,kBAAyB,EAAA;AACtC,MAAA,OAAO,oBAAoB,QAAa,KAAA,QAAA;AAAA,KAC1C;AAAA,IACA,gBAAgB,MAAuB,EAAA;AAErC,MAAO,MAAA,CAAA,MAAA,CAAO,MAAO,CAAA,IAAA,CAAK,CAAyG,uGAAA,CAAA,CAAA;AAInI,MAAqB,oBAAA,CAAA;AAAA,QACnB,MAAA;AAAA,QACA,mBAAqB,EAAA;AAAA,UACnB,QAAA,sBAAc,GAAI,EAAA;AAAA,UAClB,OAAA,sBAAa,GAAI,EAAA;AAAA,UACjB,OAAA,sBAAa,GAAI,EAAA;AAAA,UACjB,OAAA,sBAAa,GAAI,EAAA;AAAA,UACjB,QAAA,sBAAc,GAAI,EAAA;AAAA,UAClB,MAAA,sBAAY,GAAI,EAAA;AAAA,UAChB,QAAQ,EAAC;AAAA,UACT,aAAa,EAAC;AAAA,UACd,WAAa,EAAA,IAAA;AAAA,UACb,aAAa,EAAC;AAAA,UACd,cAAc,EAAC;AAAA,UACf,cAAc,EAAC;AAAA,UACf,cAAc,EAAC;AAAA;AAAA,UAEf,eAAe;AAAC,SAClB;AAAA,QACA,WAAA;AAAA,QACA,gBAAgB,EAAC;AAAA,QACjB,gBAAgB,MAAO,CAAA;AAAA,OACxB,CAAA;AAAA;AACH,GACF;AAOA,EAAA,MAAM,YAAuB,GAAA;AAAA,IAC3B,IAAM,EAAA,yCAAA;AAAA,IACN,KAAO,EAAA,OAAA;AAAA,IACP,OAAS,EAAA,MAAA;AAAA,IACT,mBAAmB,GAAU,EAAA;AAC3B,MAAA,OAAO,KAAK,IAAS,KAAA,QAAA;AAAA,KACvB;AAAA,IACA,SAAA,CAAU,MAAc,EAAY,EAAA;AAClC,MAAI,IAAA,EAAA,CAAG,QAAS,CAAA,cAAc,CAAG,EAAA;AACjC,MAAA,IAAI,CAAC,EAAA,CAAG,KAAM,CAAA,YAAY,CAAG,EAAA;AAE7B,MAAA,MAAM,aAAgB,GAAA,mDAAA;AACtB,MAAA,IAAI,CAAC,aAAA,CAAc,IAAK,CAAA,IAAI,CAAG,EAAA;AAE/B,MAAA,aAAA,CAAc,SAAY,GAAA,CAAA;AAC1B,MAAA,IAAI,OAAU,GAAA,CAAA;AACd,MAAA,MAAM,SAAS,IAAK,CAAA,OAAA,CAAQ,aAAe,EAAA,CAAC,QAAQ,OAAY,KAAA;AAC9D,QAAM,MAAA,KAAA,GAAQ,eAAe,OAAS,EAAA,CAAA,CAAA;AACtC,QAAA,OAAO,CAAU,OAAA,EAAA,KAAK,CAAyB,sBAAA,EAAA,OAAO,OAAO,KAAK,CAAA,CAAA,CAAA;AAAA,OACnE,CAAA;AAED,MAAA,OAAO,EAAE,IAAA,EAAM,MAAQ,EAAA,GAAA,EAAK,IAAK,EAAA;AAAA;AACnC,GACF;AAEA,EAAO,OAAA,CAAC,SAAW,EAAA,YAAA,EAAc,YAAY,CAAA;AAC/C;;;;"}
@@ -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;AAyCrB,eAAO,MAAM,iBAAiB,EAAE,YA+nB/B,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 { renderPages } from "./renderPages.js";
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;AA0GzF;;GAEG;AACH,eAAO,MAAM,kBAAkB,EAAE,aAkJhC,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
- // Write files for success and skip results
49
- if ((result.type === "success" || result.type === "skip") && result.html && result.rsc) {
50
- const rscWritePromise = fileWriter(result.rsc, "rsc", { ...options, route, logger: options.logger }, options.signal);
51
- const htmlWritePromise = fileWriter(result.html, "html", { ...options, route, logger: options.logger }, options.signal);
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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"}
@@ -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;