vite-plugin-react-server 1.3.6 → 1.4.1

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 (56) hide show
  1. package/README.md +33 -19
  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/configureReactServer.client.js +1 -1
  6. package/dist/plugin/dev-server/plugin.client.d.ts.map +1 -1
  7. package/dist/plugin/dev-server/plugin.client.js +17 -15
  8. package/dist/plugin/dev-server/plugin.server.d.ts.map +1 -1
  9. package/dist/plugin/dev-server/plugin.server.js +54 -4
  10. package/dist/plugin/helpers/requestInfo.d.ts.map +1 -1
  11. package/dist/plugin/helpers/requestInfo.js +4 -3
  12. package/dist/plugin/helpers/requestToRoute.d.ts.map +1 -1
  13. package/dist/plugin/helpers/requestToRoute.js +2 -2
  14. package/dist/plugin/react-static/plugin.server.d.ts.map +1 -1
  15. package/dist/plugin/react-static/plugin.server.js +9 -1
  16. package/dist/plugin/react-static/renderPagesBatched.d.ts.map +1 -1
  17. package/dist/plugin/react-static/renderPagesBatched.js +136 -36
  18. package/dist/plugin/react-static/types.d.ts +1 -0
  19. package/dist/plugin/react-static/types.d.ts.map +1 -1
  20. package/dist/plugin/types.d.ts +15 -0
  21. package/dist/plugin/types.d.ts.map +1 -1
  22. package/dist/plugin/utils/createReactFetcher.js +24 -2
  23. package/dist/plugin/utils/useRscHmr.js +10 -2
  24. package/dist/plugin/vendor/register-vendor.js +1 -1
  25. package/dist/plugin/vendor/vendor-alias.d.ts +5 -2
  26. package/dist/plugin/vendor/vendor-alias.d.ts.map +1 -1
  27. package/dist/plugin/vendor/vendor-alias.js +60 -26
  28. package/dist/plugin/vendor/vendor.server.d.ts.map +1 -1
  29. package/dist/plugin/vendor/vendor.server.js +2 -2
  30. package/dist/plugin/vendor/vendor.static.d.ts.map +1 -1
  31. package/dist/plugin/vendor/vendor.static.js +2 -2
  32. package/dist/tsconfig.tsbuildinfo +1 -1
  33. package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-client.browser.development.js +1 -1
  34. package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-client.browser.production.js +1 -1
  35. package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-client.node.development.js +1 -1
  36. package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-client.node.production.js +1 -1
  37. package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-server.node.development.js +1 -1
  38. package/oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-server.node.production.js +1 -1
  39. package/oss-experimental/react-server-dom-esm/esm/react-server-dom-esm-client.browser.development.js +1 -1
  40. package/oss-experimental/react-server-dom-esm/esm/react-server-dom-esm-client.browser.production.js +1 -1
  41. package/oss-experimental/react-server-dom-esm/package.json +3 -3
  42. package/package.json +4 -2
  43. package/plugin/config/resolveOptions.ts +2 -0
  44. package/plugin/dev-server/configureReactServer.client.ts +1 -1
  45. package/plugin/dev-server/plugin.client.ts +23 -20
  46. package/plugin/dev-server/plugin.server.ts +71 -4
  47. package/plugin/helpers/requestInfo.ts +6 -3
  48. package/plugin/helpers/requestToRoute.ts +3 -2
  49. package/plugin/react-static/plugin.server.ts +11 -1
  50. package/plugin/react-static/renderPagesBatched.ts +148 -39
  51. package/plugin/react-static/types.ts +1 -0
  52. package/plugin/types.ts +15 -0
  53. package/plugin/vendor/register-vendor.ts +1 -1
  54. package/plugin/vendor/vendor-alias.ts +61 -39
  55. package/plugin/vendor/vendor.server.ts +3 -3
  56. package/plugin/vendor/vendor.static.ts +3 -2
@@ -3,6 +3,7 @@
3
3
  * Copyright (c) Nico Brinkkemper
4
4
  * MIT License
5
5
  */
6
+ import { readFileSync } from 'node:fs';
6
7
  import { configureReactServer } from './configureReactServer.client.js';
7
8
  import { resolveOptions } from '../config/resolveOptions.js';
8
9
 
@@ -27,8 +28,7 @@ const vitePluginReactDevServer = function _vitePluginReactServerDevClient(option
27
28
  name: "vite-plugin-react-server:dev-server-client",
28
29
  apply: "serve",
29
30
  // Only apply in dev server mode
30
- // Note: Removed applyToEnvironment - handleHotUpdate needs to run regardless
31
- // The plugin should apply to client environment, but handleHotUpdate is a dev server hook
31
+ // No applyToEnvironment hotUpdate needs to run for all environments
32
32
  // that should work regardless of environment filtering
33
33
  config(_config, viteConfigEnv) {
34
34
  configEnv = viteConfigEnv;
@@ -60,28 +60,31 @@ const vitePluginReactDevServer = function _vitePluginReactServerDevClient(option
60
60
  resolvedConfig: server.config
61
61
  });
62
62
  },
63
- handleHotUpdate({ file, server }) {
63
+ hotUpdate(ctx) {
64
+ const { file, server } = ctx;
65
+ const envName = ctx.environment?.name ?? "unknown";
66
+ if (envName !== "client") return;
64
67
  if (isProcessingHmr) {
65
68
  return void 0;
66
69
  }
67
70
  const moduleBase = userOptions.moduleBase || "src";
68
71
  const projectRoot = userOptions.projectRoot || server.config.root;
69
72
  const normalizedFile = file.replace(projectRoot, "").replace(/^\/+/, "");
70
- const isServerFile = normalizedFile.startsWith(moduleBase + "/") && (file.endsWith(".tsx") || file.endsWith(".ts") || file.endsWith(".jsx") || file.endsWith(".js"));
71
- server.config.logger.info(`[vite-plugin-react-server] handleHotUpdate: file=${file}, normalized=${normalizedFile}, isServerFile=${isServerFile}, hasHandler=${!!hmrHandler}`);
73
+ const isSourceFile = normalizedFile.startsWith(moduleBase + "/") && (file.endsWith(".tsx") || file.endsWith(".ts") || file.endsWith(".jsx") || file.endsWith(".js"));
74
+ const isClientFile = isSourceFile && (() => {
75
+ try {
76
+ const head = readFileSync(file, "utf-8").slice(0, 200);
77
+ return /^\s*["']use client["']/.test(head.split("\n")[0]);
78
+ } catch {
79
+ return false;
80
+ }
81
+ })();
82
+ const isServerFile = isSourceFile && !isClientFile;
72
83
  if (isServerFile && hmrHandler) {
73
84
  isProcessingHmr = true;
74
85
  try {
75
86
  server.config.logger.info(`[vite-plugin-react-server] File changed: ${file}, sending HMR update...`);
76
87
  hmrHandler.sendHmrUpdate(file);
77
- server.ws.send({
78
- type: "custom",
79
- event: "vite-plugin-react-server:server-component-update",
80
- data: {
81
- file: normalizedFile,
82
- path: file
83
- }
84
- });
85
88
  if (hmrHandler.restart) {
86
89
  if (restartTimeout) {
87
90
  clearTimeout(restartTimeout);
@@ -113,10 +116,9 @@ const vitePluginReactDevServer = function _vitePluginReactServerDevClient(option
113
116
  } else if (isServerFile && !hmrHandler) {
114
117
  server.config.logger.warn(`[vite-plugin-react-server] Server file changed but HMR handler not available yet: ${file}`);
115
118
  }
116
- return void 0;
117
119
  }
118
120
  };
119
121
  };
120
122
 
121
123
  export { vitePluginReactDevServer };
122
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"plugin.client.js","sources":["../../../plugin/dev-server/plugin.client.ts"],"sourcesContent":["import type { VitePluginFn } from \"../../types.js\";\nimport { configureReactServer } from \"./configureReactServer.client.js\";\nimport { resolveOptions } from \"../config/resolveOptions.js\";\nimport type { ConfigEnv } from \"vite\";\n\n\n/**\n * Dev server plugin for client environment.\n * Uses configureServer hook for proper dev server setup.\n */\nexport const vitePluginReactDevServer: VitePluginFn = function _vitePluginReactServerDevClient(options) {\n  if (options == null) {\n    throw new Error(\"options is required\");\n  }\n\n  if (options.verbose) {\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\n  let configEnv: ConfigEnv | undefined;\n  let hmrHandler: { restart: (() => Promise<void>) | null; sendHmrUpdate: (file: string, routes?: string[]) => void } | null = null;\n  let isProcessingHmr = false; // Prevent recursive HMR updates\n  let restartTimeout: NodeJS.Timeout | null = null; // Debounce worker restarts\n\n  return {\n    name: \"vite-plugin-react-server:dev-server-client\",\n    apply: \"serve\", // Only apply in dev server mode\n    // Note: Removed applyToEnvironment - handleHotUpdate needs to run regardless\n    // The plugin should apply to client environment, but handleHotUpdate is a dev server hook\n    // that should work regardless of environment filtering\n    config(_config, viteConfigEnv) {\n      configEnv = viteConfigEnv;\n  \n    },\n    configureServer(server) {      \n      // Log that plugin is being configured\n      server.config.logger.info(`[vite-plugin-react-server] Dev server plugin configured for client environment`);\n      \n      // Configure the React server for client environment (worker-based)\n      // This uses the existing configureReactServer.client.js implementation\n      hmrHandler = 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        configEnv: configEnv!,\n        serverManifest: {}, \n        resolvedConfig: server.config,\n      });\n    },\n    handleHotUpdate({ file, server }) {\n      // Prevent recursive HMR updates\n      if (isProcessingHmr) {\n        return undefined;\n      }\n      \n      // Handle server component file changes\n      const moduleBase = userOptions.moduleBase || \"src\";\n      const projectRoot = userOptions.projectRoot || server.config.root;\n      \n      // Normalize paths for comparison (handle both absolute and relative)\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      // Always log for debugging\n      server.config.logger.info(`[vite-plugin-react-server] handleHotUpdate: file=${file}, normalized=${normalizedFile}, isServerFile=${isServerFile}, hasHandler=${!!hmrHandler}`);\n      \n      if (isServerFile && hmrHandler) {\n        isProcessingHmr = true;\n        \n        try {\n          server.config.logger.info(`[vite-plugin-react-server] File changed: ${file}, sending HMR update...`);\n          \n          // CRITICAL: Send HMR_UPDATE message to worker to invalidate modules\n          // This clears component caches, but Node.js's ES module cache persists\n          hmrHandler.sendHmrUpdate(file);\n          \n          // CRITICAL: Send custom HMR message to client via WebSocket\n          // Server components aren't in the client bundle, so vite:beforeUpdate doesn't fire\n          // We need to manually notify the client to refetch the 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          // CRITICAL: Node.js caches ES modules, so we need to restart the worker\n          // to clear the module cache. Debounce restarts to prevent recursion.\n          if (hmrHandler.restart) {\n            // Clear any pending restart\n            if (restartTimeout) {\n              clearTimeout(restartTimeout);\n            }\n            \n            // Debounce worker restart to prevent recursion\n            // Wait 500ms after the last file change before restarting\n            restartTimeout = setTimeout(async () => {\n              try {\n                server.config.logger.info(`[vite-plugin-react-server] Restarting worker to clear Node.js module cache...`);\n                await hmrHandler!.restart!();\n                server.config.logger.info(`[vite-plugin-react-server] Worker restarted successfully`);\n              } catch (error) {\n                server.config.logger.error(`[vite-plugin-react-server] Failed to restart worker: ${error}`);\n              } finally {\n                restartTimeout = null;\n                // Reset flag after restart completes\n                setTimeout(() => {\n                  isProcessingHmr = false;\n                }, 100);\n              }\n            }, 500); // 500ms debounce\n          } else {\n            // No restart function available yet - worker hasn't been created\n            // This is expected if handleHotUpdate fires before the first request\n            server.config.logger.warn(`[vite-plugin-react-server] Restart function not available yet - worker will be created on next request`);\n            setTimeout(() => {\n              isProcessingHmr = false;\n            }, 100);\n          }\n        } catch (error) {\n          server.config.logger.error(`[vite-plugin-react-server] Error handling HMR update: ${error}`);\n          isProcessingHmr = false;\n        }\n      } else if (isServerFile && !hmrHandler) {\n        server.config.logger.warn(`[vite-plugin-react-server] Server file changed but HMR handler not available yet: ${file}`);\n      }\n      \n      // Return undefined to allow other plugins to handle the update\n      return undefined;\n    },\n  };\n};\n"],"names":[],"mappings":"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"}
124
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"plugin.client.js","sources":["../../../plugin/dev-server/plugin.client.ts"],"sourcesContent":["import { readFileSync } from \"node:fs\";\nimport type { VitePluginFn } from \"../../types.js\";\nimport { configureReactServer } from \"./configureReactServer.client.js\";\nimport { resolveOptions } from \"../config/resolveOptions.js\";\nimport type { ConfigEnv } from \"vite\";\n\n\n/**\n * Dev server plugin for client environment.\n * Uses configureServer hook for proper dev server setup.\n */\nexport const vitePluginReactDevServer: VitePluginFn = function _vitePluginReactServerDevClient(options) {\n  if (options == null) {\n    throw new Error(\"options is required\");\n  }\n\n  if (options.verbose) {\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\n  let configEnv: ConfigEnv | undefined;\n  let hmrHandler: { restart: (() => Promise<void>) | null; sendHmrUpdate: (file: string, routes?: string[]) => void } | null = null;\n  let isProcessingHmr = false; // Prevent recursive HMR updates\n  let restartTimeout: NodeJS.Timeout | null = null; // Debounce worker restarts\n\n  return {\n    name: \"vite-plugin-react-server:dev-server-client\",\n    apply: \"serve\", // Only apply in dev server mode\n    // No applyToEnvironment — hotUpdate needs to run for all environments\n    // that should work regardless of environment filtering\n    config(_config, viteConfigEnv) {\n      configEnv = viteConfigEnv;\n  \n    },\n    configureServer(server) {      \n      // Log that plugin is being configured\n      server.config.logger.info(`[vite-plugin-react-server] Dev server plugin configured for client environment`);\n      \n      // Configure the React server for client environment (worker-based)\n      // This uses the existing configureReactServer.client.js implementation\n      hmrHandler = 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        configEnv: configEnv!,\n        serverManifest: {}, \n        resolvedConfig: server.config,\n      });\n    },\n    hotUpdate(ctx: any) {\n      const { file, server } = ctx;\n      const envName = ctx.environment?.name ?? 'unknown';\n      \n      // Only run worker invalidation from the client environment (once per change)\n      if (envName !== 'client') return;\n      \n      // Prevent recursive HMR updates\n      if (isProcessingHmr) {\n        return undefined;\n      }\n      \n      // Handle server component file changes\n      const moduleBase = userOptions.moduleBase || \"src\";\n      const projectRoot = userOptions.projectRoot || server.config.root;\n      \n      // Normalize paths for comparison (handle both absolute and relative)\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 with Fast Refresh\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      const isServerFile = isSourceFile && !isClientFile;\n      \n      if (isServerFile && hmrHandler) {\n        isProcessingHmr = true;\n        \n        try {\n          server.config.logger.info(`[vite-plugin-react-server] File changed: ${file}, sending HMR update...`);\n          \n          // CRITICAL: Send HMR_UPDATE message to worker to invalidate modules\n          // This clears component caches, but Node.js's ES module cache persists\n          hmrHandler.sendHmrUpdate(file);\n          \n          // NOTE: The WS event to notify the client is sent by the hotUpdate hook\n          // in plugin.server.ts — don't duplicate it here.\n          \n          // CRITICAL: Node.js caches ES modules, so we need to restart the worker\n          // to clear the module cache. Debounce restarts to prevent recursion.\n          if (hmrHandler.restart) {\n            // Clear any pending restart\n            if (restartTimeout) {\n              clearTimeout(restartTimeout);\n            }\n            \n            // Debounce worker restart to prevent recursion\n            // Wait 500ms after the last file change before restarting\n            restartTimeout = setTimeout(async () => {\n              try {\n                server.config.logger.info(`[vite-plugin-react-server] Restarting worker to clear Node.js module cache...`);\n                await hmrHandler!.restart!();\n                server.config.logger.info(`[vite-plugin-react-server] Worker restarted successfully`);\n              } catch (error) {\n                server.config.logger.error(`[vite-plugin-react-server] Failed to restart worker: ${error}`);\n              } finally {\n                restartTimeout = null;\n                // Reset flag after restart completes\n                setTimeout(() => {\n                  isProcessingHmr = false;\n                }, 100);\n              }\n            }, 500); // 500ms debounce\n          } else {\n            // No restart function available yet - worker hasn't been created\n            // This is expected if hotUpdate fires before the first request\n            server.config.logger.warn(`[vite-plugin-react-server] Restart function not available yet - worker will be created on next request`);\n            setTimeout(() => {\n              isProcessingHmr = false;\n            }, 100);\n          }\n        } catch (error) {\n          server.config.logger.error(`[vite-plugin-react-server] Error handling HMR update: ${error}`);\n          isProcessingHmr = false;\n        }\n      } else if (isServerFile && !hmrHandler) {\n        server.config.logger.warn(`[vite-plugin-react-server] Server file changed but HMR handler not available yet: ${file}`);\n      }\n      \n      // Don't suppress — plugin.server.ts hotUpdate handles page reload prevention\n    },\n  };\n};\n"],"names":[],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"plugin.server.d.ts","sourceRoot":"","sources":["../../../plugin/dev-server/plugin.server.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAG1D,OAAO,KAAK,EAAE,MAAM,EAAiB,MAAM,MAAM,CAAC;AAElD;;;GAGG;AACH,eAAO,MAAM,wBAAwB,GAA4C,SAAS,mBAAmB,KAAG,MAAM,EA2HrH,CAAC"}
1
+ {"version":3,"file":"plugin.server.d.ts","sourceRoot":"","sources":["../../../plugin/dev-server/plugin.server.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAG1D,OAAO,KAAK,EAAE,MAAM,EAAiB,MAAM,MAAM,CAAC;AAGlD;;;GAGG;AACH,eAAO,MAAM,wBAAwB,GAA4C,SAAS,mBAAmB,KAAG,MAAM,EA6LrH,CAAC"}
@@ -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":"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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":"requestInfo.d.ts","sourceRoot":"","sources":["../../../plugin/helpers/requestInfo.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,aAAa,CAAC;AACxD,OAAO,EAAE,KAAK,OAAO,EAAE,MAAM,MAAM,CAAC;AAKpC;;;;;;;;;GASG;AACH,wBAAgB,WAAW,CACzB,GAAG,EAAE,OAAO,CAAC,eAAe,EAC5B,cAAc,EAAE,IAAI,CAClB,oBAAoB,EAClB,YAAY,GACZ,OAAO,GACP,cAAc,GACd,SAAS,GACT,gBAAgB,GAChB,eAAe,GACf,SAAS,GACT,QAAQ,CACX,EACD,OAAO,EAAE,MAAM;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EA0MhB"}
1
+ {"version":3,"file":"requestInfo.d.ts","sourceRoot":"","sources":["../../../plugin/helpers/requestInfo.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,aAAa,CAAC;AACxD,OAAO,EAAE,KAAK,OAAO,EAAE,MAAM,MAAM,CAAC;AAKpC;;;;;;;;;GASG;AACH,wBAAgB,WAAW,CACzB,GAAG,EAAE,OAAO,CAAC,eAAe,EAC5B,cAAc,EAAE,IAAI,CAClB,oBAAoB,EAClB,YAAY,GACZ,OAAO,GACP,cAAc,GACd,SAAS,GACT,gBAAgB,GAChB,eAAe,GACf,SAAS,GACT,QAAQ,CACX,EACD,OAAO,EAAE,MAAM;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EA6MhB"}
@@ -47,6 +47,7 @@ function requestInfo(req, handlerOptions, hostDir) {
47
47
  const hasJsonHeader = req.headers["accept"]?.includes("application/json");
48
48
  const hasHtmlHeader = req.headers.accept?.includes("text/html");
49
49
  const hasRscHeader = req.headers.accept?.includes("text/x-component");
50
+ const hasRscQueryParam = /[?&]_rsc\b/.test(req.url || "");
50
51
  const hasCssHeader = req.headers.accept?.includes("text/css");
51
52
  const isFolder = !ext;
52
53
  const isFormContentType = req.headers["content-type"]?.includes(
@@ -58,8 +59,8 @@ function requestInfo(req, handlerOptions, hostDir) {
58
59
  const isFormActionRequest = !isServerActionRequest && (req.method === "POST" || isFormContentType && req.headers["sec-fetch-dest"] === "document" && req.headers["sec-fetch-mode"] === "navigate");
59
60
  const isJsRequest = !isFormActionRequest && !isJson && !isHtml && !isCss && !isRsc && (isJS || hasJsHeader);
60
61
  const isJsonRequest = isJson || hasJsonHeader && !isJsRequest;
61
- const isHtmlRequest = isHtml || hasHtmlHeader || isFolder && !hasRscHeader && !isRsc && !isJsRequest && !isFormActionRequest;
62
- const isRscRequest = !isJsRequest && !isHtmlRequest && (isRsc || hasRscHeader);
62
+ const isHtmlRequest = !hasRscQueryParam && (isHtml || hasHtmlHeader || isFolder && !hasRscHeader && !hasRscQueryParam && !isRsc && !isJsRequest && !isFormActionRequest);
63
+ const isRscRequest = hasRscQueryParam || !isJsRequest && !isHtmlRequest && (isRsc || hasRscHeader);
63
64
  const isCssRequest = !isHtmlRequest && !isRscRequest && !isJsRequest && !isJsonRequest && (isCss || hasCssHeader);
64
65
  const routeForFilePath = value;
65
66
  let filePath = resolve(hostDir, routeForFilePath);
@@ -159,4 +160,4 @@ function requestInfo(req, handlerOptions, hostDir) {
159
160
  }
160
161
 
161
162
  export { requestInfo };
162
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"requestInfo.js","sources":["../../../plugin/helpers/requestInfo.ts"],"sourcesContent":["import { resolve } from \"node:path\";\nimport type { CreateHandlerOptions } from \"../types.js\";\nimport { type Connect } from \"vite\";\nimport { MIME_TYPES } from \"../config/mimeTypes.js\";\nimport { requestToRoute } from \"./requestToRoute.js\";\nimport { routeToURL } from \"../utils/routeToURL.js\";\n\n/**\n * # Request info\n *\n * Does the initial work to check if the request is for html, rsc, json, js, css, server-action, or something else not handled by this plugin.\n *\n * @param req\n * @param handlerOptions\n * @param hostDir\n * @returns\n */\nexport function requestInfo(\n  req: Connect.IncomingMessage,\n  handlerOptions: Pick<\n    CreateHandlerOptions,\n    | \"normalizer\"\n    | \"build\"\n    | \"autoDiscover\"\n    | \"verbose\"\n    | \"moduleBasePath\"\n    | \"moduleBaseURL\"\n    | \"verbose\"\n    | \"logger\"\n  >,\n  hostDir: string,\n) {\n  const route = requestToRoute(req, {\n    moduleBasePath: handlerOptions.moduleBasePath,\n    moduleBaseURL: handlerOptions.moduleBaseURL,\n    build: handlerOptions.build,\n  });\n\n  if (!route) {\n    return {\n      route: \"/\",\n      url: routeToURL(\"/\", handlerOptions.moduleBaseURL, handlerOptions.build.rscOutputPath),\n      ext: \"\",\n    };\n  }\n\n  // Use the cleaned route for normalization, not the raw req.url\n  // This ensures base URL is properly stripped before normalization\n  const [, value] = handlerOptions.normalizer(route);\n  if (handlerOptions.verbose) {\n    if (value && value !== \"\") {\n      handlerOptions.logger.info(`[requestInfo] Value: \\\"${value}\\\"`);\n    }\n    if (hostDir && hostDir !== \"\") {\n      handlerOptions.logger.info(`[requestInfo] Host Dir: \\\"${hostDir}\\\"`);\n    }\n    if (req.url && req.url !== \"\") {\n      handlerOptions.logger.info(`[requestInfo] Request URL: \\\"${req.url}\\\"`);\n    }\n  }\n\n  const dotIndex = value.lastIndexOf(\".\");\n  const ext = dotIndex === -1 ? \"\" : value.slice(dotIndex);\n  // handle index.html\n  const isVendor = handlerOptions.autoDiscover.vendorPattern.test(value);\n  const isVirtual = handlerOptions.autoDiscover.virtualPattern.test(value);\n  const isJS = handlerOptions.autoDiscover.modulePattern.test(value);\n  const isHtml = handlerOptions.autoDiscover.htmlPattern.test(value);\n  const isCss = handlerOptions.autoDiscover.cssPattern.test(value);\n  const isJson = handlerOptions.autoDiscover.jsonPattern.test(value);\n  const isRsc = handlerOptions.autoDiscover.rscPattern.test(value);\n  const hasJsHeader =\n    req.headers[\"sec-fetch-dest\"] === \"script\" ||\n    req.headers[\"accept\"]?.includes(\"*/*\") ||\n    req.headers[\"accept\"]?.includes(\"text/javascript\");\n  const hasJsonHeader = req.headers[\"accept\"]?.includes(\"application/json\");\n  const hasHtmlHeader = req.headers.accept?.includes(\"text/html\");\n  const hasRscHeader = req.headers.accept?.includes(\"text/x-component\");\n  const hasCssHeader = req.headers.accept?.includes(\"text/css\");\n  const isFolder = !ext;\n  const isFormContentType =\n    req.headers[\"content-type\"]?.includes(\n      \"application/x-www-form-urlencoded\"\n    ) || !!req.headers[\"content-type\"]?.includes(\"multipart/form-data\");\n\n  // Server action detection\n  const hasRscActionHeader = !!req.headers[\"x-rsc-action\"];\n  const hasServerActionHeaders =\n    req.method === \"POST\" &&\n    (hasRscActionHeader || (\n      (req.headers[\"sec-fetch-dest\"] === \"empty\" ||\n        req.headers[\"sec-fetch-dest\"] === \"\") &&\n      req.headers[\"sec-fetch-mode\"] === \"cors\"\n    ));\n  const isServerActionRequest = hasServerActionHeaders;\n\n  const isFormActionRequest =\n    !isServerActionRequest &&\n    (req.method === \"POST\" ||\n      (isFormContentType &&\n        req.headers[\"sec-fetch-dest\"] === \"document\" &&\n        req.headers[\"sec-fetch-mode\"] === \"navigate\"));\n\n  const isJsRequest =\n    !isFormActionRequest &&\n    !isJson &&\n    !isHtml &&\n    !isCss &&\n    !isRsc &&\n    (isJS || hasJsHeader);\n  const isJsonRequest = isJson || (hasJsonHeader && !isJsRequest);\n  // Form action detection\n\n  const isHtmlRequest =\n    isHtml ||\n    hasHtmlHeader ||\n    (isFolder &&\n      !hasRscHeader &&\n      !isRsc &&\n      !isJsRequest &&\n      !isFormActionRequest);\n  const isRscRequest =\n    !isJsRequest && !isHtmlRequest && (isRsc || hasRscHeader);\n  const isCssRequest =\n    !isHtmlRequest &&\n    !isRscRequest &&\n    !isJsRequest &&\n    !isJsonRequest &&\n    (isCss || hasCssHeader);\n\n  // Use the normalized value for file path construction\n  // The normalizer should have already stripped base URLs properly\n  const routeForFilePath = value;\n\n  let filePath = resolve(hostDir, routeForFilePath);\n  let contentType;\n  if (isServerActionRequest) {\n    // For server actions, we'll get the actual file path from the request body\n    // The route is just a placeholder\n    filePath = resolve(hostDir, routeForFilePath);\n    contentType = \"application/json; charset=utf-8\";\n  } else if (isHtmlRequest) {\n    if (!isHtml) {\n      filePath = resolve(\n        hostDir,\n        routeForFilePath,\n        handlerOptions.build.htmlOutputPath\n      );\n    }\n    contentType = \"text/html; charset=utf-8\";\n  } else if (isRscRequest) {\n    if (!isRsc) {\n      // Value doesn't end with .rsc, append the rsc output path\n      filePath = resolve(\n        hostDir,\n        routeForFilePath,\n        handlerOptions.build.rscOutputPath\n      );\n    }\n    contentType = \"text/x-component; charset=utf-8\";\n  } else if (isCssRequest) {\n    if (!isCss) {\n      filePath = resolve(hostDir, routeForFilePath + \".css\");\n    }\n    contentType = \"text/css; charset=utf-8\";\n  } else if (isJsRequest) {\n    if (!isJS) {\n      filePath = resolve(hostDir, routeForFilePath + \".js\");\n    }\n    contentType = \"application/javascript; charset=utf-8\";\n  } else if (isJsonRequest) {\n    if (!isJson) {\n      filePath = resolve(hostDir, routeForFilePath + \".json\");\n    }\n    contentType = \"application/json; charset=utf-8\";\n  } else {\n    const mimeType = MIME_TYPES[ext];\n    if (mimeType) {\n      contentType = mimeType + \"; charset=utf-8\";\n    } else {\n      contentType = \"application/octet-stream; charset=utf-8\";\n    }\n  }\n\n  if (handlerOptions.verbose) {\n    if (isFormActionRequest) {\n      handlerOptions.logger.info(`[react-dev-server] (form-action) ${route}`);\n    } else if (isServerActionRequest) {\n      handlerOptions.logger.info(`[react-dev-server] (server-action) ${route}`);\n    } else if (isHtmlRequest) {\n      handlerOptions.logger.info(`[react-dev-server] (html) ${route}`);\n    } else if (isRscRequest) {\n      handlerOptions.logger.info(`[react-dev-server] (rsc) ${route}`);\n    } else if (isCssRequest) {\n      handlerOptions.logger.info(`[react-dev-server] (css) ${route}`);\n    } else if (isJsRequest) {\n      handlerOptions.logger.info(`[react-dev-server] (js) ${route}`);\n    } else if (isJsonRequest) {\n      handlerOptions.logger.info(`[react-dev-server] (json) ${route}`);\n    } else {\n      handlerOptions.logger.info(`[react-dev-server] (other) ${route}`);\n    }\n  }\n  return {\n    route,\n    url: routeToURL(route, handlerOptions.moduleBaseURL, handlerOptions.build.rscOutputPath),\n    ext,\n    isHtmlRequest,\n    isRscRequest,\n    isCssRequest,\n    isCss,\n    isHtml,\n    isRsc,\n    isFolder,\n    contentType,\n    filePath,\n    isJS,\n    isVendor,\n    isVirtual,\n    hasJsHeader,\n    isJsRequest,\n    isJson,\n    isJsonRequest,\n    hasCssHeader,\n    hasJsonHeader,\n    hasHtmlHeader,\n    hasRscHeader,\n    hasServerActionHeaders,\n    isServerActionRequest,\n    isFormContentType,\n    isFormActionRequest,\n  };\n}\n"],"names":[],"mappings":"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"}
163
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"requestInfo.js","sources":["../../../plugin/helpers/requestInfo.ts"],"sourcesContent":["import { resolve } from \"node:path\";\nimport type { CreateHandlerOptions } from \"../types.js\";\nimport { type Connect } from \"vite\";\nimport { MIME_TYPES } from \"../config/mimeTypes.js\";\nimport { requestToRoute } from \"./requestToRoute.js\";\nimport { routeToURL } from \"../utils/routeToURL.js\";\n\n/**\n * # Request info\n *\n * Does the initial work to check if the request is for html, rsc, json, js, css, server-action, or something else not handled by this plugin.\n *\n * @param req\n * @param handlerOptions\n * @param hostDir\n * @returns\n */\nexport function requestInfo(\n  req: Connect.IncomingMessage,\n  handlerOptions: Pick<\n    CreateHandlerOptions,\n    | \"normalizer\"\n    | \"build\"\n    | \"autoDiscover\"\n    | \"verbose\"\n    | \"moduleBasePath\"\n    | \"moduleBaseURL\"\n    | \"verbose\"\n    | \"logger\"\n  >,\n  hostDir: string,\n) {\n  const route = requestToRoute(req, {\n    moduleBasePath: handlerOptions.moduleBasePath,\n    moduleBaseURL: handlerOptions.moduleBaseURL,\n    build: handlerOptions.build,\n  });\n\n  if (!route) {\n    return {\n      route: \"/\",\n      url: routeToURL(\"/\", handlerOptions.moduleBaseURL, handlerOptions.build.rscOutputPath),\n      ext: \"\",\n    };\n  }\n\n  // Use the cleaned route for normalization, not the raw req.url\n  // This ensures base URL is properly stripped before normalization\n  const [, value] = handlerOptions.normalizer(route);\n  if (handlerOptions.verbose) {\n    if (value && value !== \"\") {\n      handlerOptions.logger.info(`[requestInfo] Value: \\\"${value}\\\"`);\n    }\n    if (hostDir && hostDir !== \"\") {\n      handlerOptions.logger.info(`[requestInfo] Host Dir: \\\"${hostDir}\\\"`);\n    }\n    if (req.url && req.url !== \"\") {\n      handlerOptions.logger.info(`[requestInfo] Request URL: \\\"${req.url}\\\"`);\n    }\n  }\n\n  const dotIndex = value.lastIndexOf(\".\");\n  const ext = dotIndex === -1 ? \"\" : value.slice(dotIndex);\n  // handle index.html\n  const isVendor = handlerOptions.autoDiscover.vendorPattern.test(value);\n  const isVirtual = handlerOptions.autoDiscover.virtualPattern.test(value);\n  const isJS = handlerOptions.autoDiscover.modulePattern.test(value);\n  const isHtml = handlerOptions.autoDiscover.htmlPattern.test(value);\n  const isCss = handlerOptions.autoDiscover.cssPattern.test(value);\n  const isJson = handlerOptions.autoDiscover.jsonPattern.test(value);\n  const isRsc = handlerOptions.autoDiscover.rscPattern.test(value);\n  const hasJsHeader =\n    req.headers[\"sec-fetch-dest\"] === \"script\" ||\n    req.headers[\"accept\"]?.includes(\"*/*\") ||\n    req.headers[\"accept\"]?.includes(\"text/javascript\");\n  const hasJsonHeader = req.headers[\"accept\"]?.includes(\"application/json\");\n  const hasHtmlHeader = req.headers.accept?.includes(\"text/html\");\n  const hasRscHeader = req.headers.accept?.includes(\"text/x-component\");\n  // Support ?_rsc query param (e.g. /?_rsc for browser debugging) as alternative to Accept header (useful for browser debugging)\n  const hasRscQueryParam = /[?&]_rsc\\b/.test(req.url || \"\");\n  const hasCssHeader = req.headers.accept?.includes(\"text/css\");\n  const isFolder = !ext;\n  const isFormContentType =\n    req.headers[\"content-type\"]?.includes(\n      \"application/x-www-form-urlencoded\"\n    ) || !!req.headers[\"content-type\"]?.includes(\"multipart/form-data\");\n\n  // Server action detection\n  const hasRscActionHeader = !!req.headers[\"x-rsc-action\"];\n  const hasServerActionHeaders =\n    req.method === \"POST\" &&\n    (hasRscActionHeader || (\n      (req.headers[\"sec-fetch-dest\"] === \"empty\" ||\n        req.headers[\"sec-fetch-dest\"] === \"\") &&\n      req.headers[\"sec-fetch-mode\"] === \"cors\"\n    ));\n  const isServerActionRequest = hasServerActionHeaders;\n\n  const isFormActionRequest =\n    !isServerActionRequest &&\n    (req.method === \"POST\" ||\n      (isFormContentType &&\n        req.headers[\"sec-fetch-dest\"] === \"document\" &&\n        req.headers[\"sec-fetch-mode\"] === \"navigate\"));\n\n  const isJsRequest =\n    !isFormActionRequest &&\n    !isJson &&\n    !isHtml &&\n    !isCss &&\n    !isRsc &&\n    (isJS || hasJsHeader);\n  const isJsonRequest = isJson || (hasJsonHeader && !isJsRequest);\n  // Form action detection\n\n  const isHtmlRequest =\n    !hasRscQueryParam && (isHtml ||\n    hasHtmlHeader ||\n    (isFolder &&\n      !hasRscHeader &&\n      !hasRscQueryParam &&\n      !isRsc &&\n      !isJsRequest &&\n      !isFormActionRequest));\n  const isRscRequest =\n    hasRscQueryParam || (!isJsRequest && !isHtmlRequest && (isRsc || hasRscHeader));\n  const isCssRequest =\n    !isHtmlRequest &&\n    !isRscRequest &&\n    !isJsRequest &&\n    !isJsonRequest &&\n    (isCss || hasCssHeader);\n\n  // Use the normalized value for file path construction\n  // The normalizer should have already stripped base URLs properly\n  const routeForFilePath = value;\n\n  let filePath = resolve(hostDir, routeForFilePath);\n  let contentType;\n  if (isServerActionRequest) {\n    // For server actions, we'll get the actual file path from the request body\n    // The route is just a placeholder\n    filePath = resolve(hostDir, routeForFilePath);\n    contentType = \"application/json; charset=utf-8\";\n  } else if (isHtmlRequest) {\n    if (!isHtml) {\n      filePath = resolve(\n        hostDir,\n        routeForFilePath,\n        handlerOptions.build.htmlOutputPath\n      );\n    }\n    contentType = \"text/html; charset=utf-8\";\n  } else if (isRscRequest) {\n    if (!isRsc) {\n      // Value doesn't end with .rsc, append the rsc output path\n      filePath = resolve(\n        hostDir,\n        routeForFilePath,\n        handlerOptions.build.rscOutputPath\n      );\n    }\n    contentType = \"text/x-component; charset=utf-8\";\n  } else if (isCssRequest) {\n    if (!isCss) {\n      filePath = resolve(hostDir, routeForFilePath + \".css\");\n    }\n    contentType = \"text/css; charset=utf-8\";\n  } else if (isJsRequest) {\n    if (!isJS) {\n      filePath = resolve(hostDir, routeForFilePath + \".js\");\n    }\n    contentType = \"application/javascript; charset=utf-8\";\n  } else if (isJsonRequest) {\n    if (!isJson) {\n      filePath = resolve(hostDir, routeForFilePath + \".json\");\n    }\n    contentType = \"application/json; charset=utf-8\";\n  } else {\n    const mimeType = MIME_TYPES[ext];\n    if (mimeType) {\n      contentType = mimeType + \"; charset=utf-8\";\n    } else {\n      contentType = \"application/octet-stream; charset=utf-8\";\n    }\n  }\n\n  if (handlerOptions.verbose) {\n    if (isFormActionRequest) {\n      handlerOptions.logger.info(`[react-dev-server] (form-action) ${route}`);\n    } else if (isServerActionRequest) {\n      handlerOptions.logger.info(`[react-dev-server] (server-action) ${route}`);\n    } else if (isHtmlRequest) {\n      handlerOptions.logger.info(`[react-dev-server] (html) ${route}`);\n    } else if (isRscRequest) {\n      handlerOptions.logger.info(`[react-dev-server] (rsc) ${route}`);\n    } else if (isCssRequest) {\n      handlerOptions.logger.info(`[react-dev-server] (css) ${route}`);\n    } else if (isJsRequest) {\n      handlerOptions.logger.info(`[react-dev-server] (js) ${route}`);\n    } else if (isJsonRequest) {\n      handlerOptions.logger.info(`[react-dev-server] (json) ${route}`);\n    } else {\n      handlerOptions.logger.info(`[react-dev-server] (other) ${route}`);\n    }\n  }\n  return {\n    route,\n    url: routeToURL(route, handlerOptions.moduleBaseURL, handlerOptions.build.rscOutputPath),\n    ext,\n    isHtmlRequest,\n    isRscRequest,\n    isCssRequest,\n    isCss,\n    isHtml,\n    isRsc,\n    isFolder,\n    contentType,\n    filePath,\n    isJS,\n    isVendor,\n    isVirtual,\n    hasJsHeader,\n    isJsRequest,\n    isJson,\n    isJsonRequest,\n    hasCssHeader,\n    hasJsonHeader,\n    hasHtmlHeader,\n    hasRscHeader,\n    hasServerActionHeaders,\n    isServerActionRequest,\n    isFormContentType,\n    isFormActionRequest,\n  };\n}\n"],"names":[],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"requestToRoute.d.ts","sourceRoot":"","sources":["../../../plugin/helpers/requestToRoute.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,aAAa,CAAC;AAExD,wBAAgB,cAAc,CAC5B,GAAG,EAAE;IAAE,GAAG,CAAC,EAAE,MAAM,CAAA;CAAE,EACrB,cAAc,EAAE,IAAI,CAClB,oBAAoB,EACpB,gBAAgB,GAAG,eAAe,GAAG,OAAO,CAC7C,sBAyCF"}
1
+ {"version":3,"file":"requestToRoute.d.ts","sourceRoot":"","sources":["../../../plugin/helpers/requestToRoute.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,aAAa,CAAC;AAExD,wBAAgB,cAAc,CAC5B,GAAG,EAAE;IAAE,GAAG,CAAC,EAAE,MAAM,CAAA;CAAE,EACrB,cAAc,EAAE,IAAI,CAClB,oBAAoB,EACpB,gBAAgB,GAAG,eAAe,GAAG,OAAO,CAC7C,sBA0CF"}
@@ -4,7 +4,7 @@
4
4
  * MIT License
5
5
  */
6
6
  function requestToRoute(req, handlerOptions) {
7
- let route = req.url?.replace(
7
+ let route = req.url?.split("?")[0].replace(
8
8
  handlerOptions.moduleBaseURL + handlerOptions.build.rscOutputPath,
9
9
  ""
10
10
  ).replace(
@@ -29,4 +29,4 @@ function requestToRoute(req, handlerOptions) {
29
29
  }
30
30
 
31
31
  export { requestToRoute };
32
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVxdWVzdFRvUm91dGUuanMiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3BsdWdpbi9oZWxwZXJzL3JlcXVlc3RUb1JvdXRlLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgQ3JlYXRlSGFuZGxlck9wdGlvbnMgfSBmcm9tIFwiLi4vdHlwZXMuanNcIjtcblxuZXhwb3J0IGZ1bmN0aW9uIHJlcXVlc3RUb1JvdXRlKFxuICByZXE6IHsgdXJsPzogc3RyaW5nIH0sXG4gIGhhbmRsZXJPcHRpb25zOiBQaWNrPFxuICAgIENyZWF0ZUhhbmRsZXJPcHRpb25zLFxuICAgIFwibW9kdWxlQmFzZVBhdGhcIiB8IFwibW9kdWxlQmFzZVVSTFwiIHwgXCJidWlsZFwiXG4gID5cbikge1xuICBsZXQgcm91dGUgPSByZXEudXJsXG4gICAgPy5yZXBsYWNlKFxuICAgICAgaGFuZGxlck9wdGlvbnMubW9kdWxlQmFzZVVSTCArIGhhbmRsZXJPcHRpb25zLmJ1aWxkLnJzY091dHB1dFBhdGgsXG4gICAgICBcIlwiXG4gICAgKVxuICAgIC5yZXBsYWNlKFxuICAgICAgaGFuZGxlck9wdGlvbnMubW9kdWxlQmFzZVVSTCArIGhhbmRsZXJPcHRpb25zLmJ1aWxkLmh0bWxPdXRwdXRQYXRoLFxuICAgICAgXCJcIlxuICAgIClcbiAgICAucmVwbGFjZSgvXlxcL2luZGV4JC8sIFwiL1wiKTtcbiAgaWYgKHR5cGVvZiByb3V0ZSAhPT0gXCJzdHJpbmdcIikge1xuICAgIHJldHVybiByb3V0ZTtcbiAgfVxuICBcbiAgLy8gU3RyaXAgbW9kdWxlQmFzZVVSTCBmcm9tIHRoZSBiZWdpbm5pbmcgb2YgdGhlIHJvdXRlXG4gIGlmIChyb3V0ZS5zdGFydHNXaXRoKGhhbmRsZXJPcHRpb25zLm1vZHVsZUJhc2VVUkwpKSB7XG4gICAgcm91dGUgPSByb3V0ZS5zbGljZShoYW5kbGVyT3B0aW9ucy5tb2R1bGVCYXNlVVJMLmxlbmd0aCk7XG4gIH1cbiAgaWYocm91dGUuZW5kc1dpdGgoaGFuZGxlck9wdGlvbnMuYnVpbGQucnNjT3V0cHV0UGF0aCkpIHtcbiAgICByb3V0ZSA9IHJvdXRlLnNsaWNlKDAsIC1oYW5kbGVyT3B0aW9ucy5idWlsZC5yc2NPdXRwdXRQYXRoLmxlbmd0aCk7XG4gIH1cbiAgXG4gIGlmIChyb3V0ZS5zdGFydHNXaXRoKGhhbmRsZXJPcHRpb25zLm1vZHVsZUJhc2VQYXRoKSkge1xuICAgIHJvdXRlID0gcm91dGUuc2xpY2UoaGFuZGxlck9wdGlvbnMubW9kdWxlQmFzZVBhdGgubGVuZ3RoKTtcbiAgfVxuXG4gIGNvbnN0IHJvdXRlV2l0aG91dFRyYWlsaW5nU2xhc2ggPVxuICAgIHJvdXRlID09PSBcIlwiIHx8IHJvdXRlID09PSBcIi9cIlxuICAgICAgPyBcIi9cIlxuICAgICAgOiByb3V0ZS5lbmRzV2l0aChcIi9cIilcbiAgICAgID8gcm91dGUuc2xpY2UoMCwgLTEpXG4gICAgICA6IHJvdXRlO1xuXG4gIGNvbnN0IHJvdXRlV2l0aExlYWRpbmdTbGFzaCA9ICFyb3V0ZVdpdGhvdXRUcmFpbGluZ1NsYXNoXG4gICAgPyBcIi9cIlxuICAgIDogcm91dGVXaXRob3V0VHJhaWxpbmdTbGFzaC5zdGFydHNXaXRoKFwiL1wiKVxuICAgID8gcm91dGVXaXRob3V0VHJhaWxpbmdTbGFzaFxuICAgIDogYC8ke3JvdXRlV2l0aG91dFRyYWlsaW5nU2xhc2h9YDtcbiAgcmV0dXJuIHJvdXRlV2l0aExlYWRpbmdTbGFzaDtcbn1cbiJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUVPLFNBQVMsY0FBQSxDQUNkLEtBQ0EsY0FJQSxFQUFBO0FBQ0EsRUFBSSxJQUFBLEtBQUEsR0FBUSxJQUFJLEdBQ1osRUFBQSxPQUFBO0FBQUEsSUFDQSxjQUFBLENBQWUsYUFBZ0IsR0FBQSxjQUFBLENBQWUsS0FBTSxDQUFBLGFBQUE7QUFBQSxJQUNwRDtBQUFBLEdBRUQsQ0FBQSxPQUFBO0FBQUEsSUFDQyxjQUFBLENBQWUsYUFBZ0IsR0FBQSxjQUFBLENBQWUsS0FBTSxDQUFBLGNBQUE7QUFBQSxJQUNwRDtBQUFBLEdBQ0YsQ0FDQyxPQUFRLENBQUEsV0FBQSxFQUFhLEdBQUcsQ0FBQTtBQUMzQixFQUFJLElBQUEsT0FBTyxVQUFVLFFBQVUsRUFBQTtBQUM3QixJQUFPLE9BQUEsS0FBQTtBQUFBO0FBSVQsRUFBQSxJQUFJLEtBQU0sQ0FBQSxVQUFBLENBQVcsY0FBZSxDQUFBLGFBQWEsQ0FBRyxFQUFBO0FBQ2xELElBQUEsS0FBQSxHQUFRLEtBQU0sQ0FBQSxLQUFBLENBQU0sY0FBZSxDQUFBLGFBQUEsQ0FBYyxNQUFNLENBQUE7QUFBQTtBQUV6RCxFQUFBLElBQUcsS0FBTSxDQUFBLFFBQUEsQ0FBUyxjQUFlLENBQUEsS0FBQSxDQUFNLGFBQWEsQ0FBRyxFQUFBO0FBQ3JELElBQUEsS0FBQSxHQUFRLE1BQU0sS0FBTSxDQUFBLENBQUEsRUFBRyxDQUFDLGNBQWUsQ0FBQSxLQUFBLENBQU0sY0FBYyxNQUFNLENBQUE7QUFBQTtBQUduRSxFQUFBLElBQUksS0FBTSxDQUFBLFVBQUEsQ0FBVyxjQUFlLENBQUEsY0FBYyxDQUFHLEVBQUE7QUFDbkQsSUFBQSxLQUFBLEdBQVEsS0FBTSxDQUFBLEtBQUEsQ0FBTSxjQUFlLENBQUEsY0FBQSxDQUFlLE1BQU0sQ0FBQTtBQUFBO0FBRzFELEVBQUEsTUFBTSx5QkFDSixHQUFBLEtBQUEsS0FBVSxFQUFNLElBQUEsS0FBQSxLQUFVLE1BQ3RCLEdBQ0EsR0FBQSxLQUFBLENBQU0sUUFBUyxDQUFBLEdBQUcsQ0FDbEIsR0FBQSxLQUFBLENBQU0sS0FBTSxDQUFBLENBQUEsRUFBRyxFQUFFLENBQ2pCLEdBQUEsS0FBQTtBQUVOLEVBQU0sTUFBQSxxQkFBQSxHQUF3QixDQUFDLHlCQUFBLEdBQzNCLEdBQ0EsR0FBQSx5QkFBQSxDQUEwQixXQUFXLEdBQUcsQ0FBQSxHQUN4Qyx5QkFDQSxHQUFBLENBQUEsQ0FBQSxFQUFJLHlCQUF5QixDQUFBLENBQUE7QUFDakMsRUFBTyxPQUFBLHFCQUFBO0FBQ1Q7Ozs7In0=
32
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVxdWVzdFRvUm91dGUuanMiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3BsdWdpbi9oZWxwZXJzL3JlcXVlc3RUb1JvdXRlLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgQ3JlYXRlSGFuZGxlck9wdGlvbnMgfSBmcm9tIFwiLi4vdHlwZXMuanNcIjtcblxuZXhwb3J0IGZ1bmN0aW9uIHJlcXVlc3RUb1JvdXRlKFxuICByZXE6IHsgdXJsPzogc3RyaW5nIH0sXG4gIGhhbmRsZXJPcHRpb25zOiBQaWNrPFxuICAgIENyZWF0ZUhhbmRsZXJPcHRpb25zLFxuICAgIFwibW9kdWxlQmFzZVBhdGhcIiB8IFwibW9kdWxlQmFzZVVSTFwiIHwgXCJidWlsZFwiXG4gID5cbikge1xuICAvLyBTdHJpcCBxdWVyeSBzdHJpbmcgYmVmb3JlIHJvdXRlIG1hdGNoaW5nXG4gIGxldCByb3V0ZSA9IHJlcS51cmw/LnNwbGl0KFwiP1wiKVswXVxuICAgIC5yZXBsYWNlKFxuICAgICAgaGFuZGxlck9wdGlvbnMubW9kdWxlQmFzZVVSTCArIGhhbmRsZXJPcHRpb25zLmJ1aWxkLnJzY091dHB1dFBhdGgsXG4gICAgICBcIlwiXG4gICAgKVxuICAgIC5yZXBsYWNlKFxuICAgICAgaGFuZGxlck9wdGlvbnMubW9kdWxlQmFzZVVSTCArIGhhbmRsZXJPcHRpb25zLmJ1aWxkLmh0bWxPdXRwdXRQYXRoLFxuICAgICAgXCJcIlxuICAgIClcbiAgICAucmVwbGFjZSgvXlxcL2luZGV4JC8sIFwiL1wiKTtcbiAgaWYgKHR5cGVvZiByb3V0ZSAhPT0gXCJzdHJpbmdcIikge1xuICAgIHJldHVybiByb3V0ZTtcbiAgfVxuICBcbiAgLy8gU3RyaXAgbW9kdWxlQmFzZVVSTCBmcm9tIHRoZSBiZWdpbm5pbmcgb2YgdGhlIHJvdXRlXG4gIGlmIChyb3V0ZS5zdGFydHNXaXRoKGhhbmRsZXJPcHRpb25zLm1vZHVsZUJhc2VVUkwpKSB7XG4gICAgcm91dGUgPSByb3V0ZS5zbGljZShoYW5kbGVyT3B0aW9ucy5tb2R1bGVCYXNlVVJMLmxlbmd0aCk7XG4gIH1cbiAgaWYocm91dGUuZW5kc1dpdGgoaGFuZGxlck9wdGlvbnMuYnVpbGQucnNjT3V0cHV0UGF0aCkpIHtcbiAgICByb3V0ZSA9IHJvdXRlLnNsaWNlKDAsIC1oYW5kbGVyT3B0aW9ucy5idWlsZC5yc2NPdXRwdXRQYXRoLmxlbmd0aCk7XG4gIH1cbiAgXG4gIGlmIChyb3V0ZS5zdGFydHNXaXRoKGhhbmRsZXJPcHRpb25zLm1vZHVsZUJhc2VQYXRoKSkge1xuICAgIHJvdXRlID0gcm91dGUuc2xpY2UoaGFuZGxlck9wdGlvbnMubW9kdWxlQmFzZVBhdGgubGVuZ3RoKTtcbiAgfVxuXG4gIGNvbnN0IHJvdXRlV2l0aG91dFRyYWlsaW5nU2xhc2ggPVxuICAgIHJvdXRlID09PSBcIlwiIHx8IHJvdXRlID09PSBcIi9cIlxuICAgICAgPyBcIi9cIlxuICAgICAgOiByb3V0ZS5lbmRzV2l0aChcIi9cIilcbiAgICAgID8gcm91dGUuc2xpY2UoMCwgLTEpXG4gICAgICA6IHJvdXRlO1xuXG4gIGNvbnN0IHJvdXRlV2l0aExlYWRpbmdTbGFzaCA9ICFyb3V0ZVdpdGhvdXRUcmFpbGluZ1NsYXNoXG4gICAgPyBcIi9cIlxuICAgIDogcm91dGVXaXRob3V0VHJhaWxpbmdTbGFzaC5zdGFydHNXaXRoKFwiL1wiKVxuICAgID8gcm91dGVXaXRob3V0VHJhaWxpbmdTbGFzaFxuICAgIDogYC8ke3JvdXRlV2l0aG91dFRyYWlsaW5nU2xhc2h9YDtcbiAgcmV0dXJuIHJvdXRlV2l0aExlYWRpbmdTbGFzaDtcbn1cbiJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUVPLFNBQVMsY0FBQSxDQUNkLEtBQ0EsY0FJQSxFQUFBO0FBRUEsRUFBQSxJQUFJLFFBQVEsR0FBSSxDQUFBLEdBQUEsRUFBSyxNQUFNLEdBQUcsQ0FBQSxDQUFFLENBQUMsQ0FDOUIsQ0FBQSxPQUFBO0FBQUEsSUFDQyxjQUFBLENBQWUsYUFBZ0IsR0FBQSxjQUFBLENBQWUsS0FBTSxDQUFBLGFBQUE7QUFBQSxJQUNwRDtBQUFBLEdBRUQsQ0FBQSxPQUFBO0FBQUEsSUFDQyxjQUFBLENBQWUsYUFBZ0IsR0FBQSxjQUFBLENBQWUsS0FBTSxDQUFBLGNBQUE7QUFBQSxJQUNwRDtBQUFBLEdBQ0YsQ0FDQyxPQUFRLENBQUEsV0FBQSxFQUFhLEdBQUcsQ0FBQTtBQUMzQixFQUFJLElBQUEsT0FBTyxVQUFVLFFBQVUsRUFBQTtBQUM3QixJQUFPLE9BQUEsS0FBQTtBQUFBO0FBSVQsRUFBQSxJQUFJLEtBQU0sQ0FBQSxVQUFBLENBQVcsY0FBZSxDQUFBLGFBQWEsQ0FBRyxFQUFBO0FBQ2xELElBQUEsS0FBQSxHQUFRLEtBQU0sQ0FBQSxLQUFBLENBQU0sY0FBZSxDQUFBLGFBQUEsQ0FBYyxNQUFNLENBQUE7QUFBQTtBQUV6RCxFQUFBLElBQUcsS0FBTSxDQUFBLFFBQUEsQ0FBUyxjQUFlLENBQUEsS0FBQSxDQUFNLGFBQWEsQ0FBRyxFQUFBO0FBQ3JELElBQUEsS0FBQSxHQUFRLE1BQU0sS0FBTSxDQUFBLENBQUEsRUFBRyxDQUFDLGNBQWUsQ0FBQSxLQUFBLENBQU0sY0FBYyxNQUFNLENBQUE7QUFBQTtBQUduRSxFQUFBLElBQUksS0FBTSxDQUFBLFVBQUEsQ0FBVyxjQUFlLENBQUEsY0FBYyxDQUFHLEVBQUE7QUFDbkQsSUFBQSxLQUFBLEdBQVEsS0FBTSxDQUFBLEtBQUEsQ0FBTSxjQUFlLENBQUEsY0FBQSxDQUFlLE1BQU0sQ0FBQTtBQUFBO0FBRzFELEVBQUEsTUFBTSx5QkFDSixHQUFBLEtBQUEsS0FBVSxFQUFNLElBQUEsS0FBQSxLQUFVLE1BQ3RCLEdBQ0EsR0FBQSxLQUFBLENBQU0sUUFBUyxDQUFBLEdBQUcsQ0FDbEIsR0FBQSxLQUFBLENBQU0sS0FBTSxDQUFBLENBQUEsRUFBRyxFQUFFLENBQ2pCLEdBQUEsS0FBQTtBQUVOLEVBQU0sTUFBQSxxQkFBQSxHQUF3QixDQUFDLHlCQUFBLEdBQzNCLEdBQ0EsR0FBQSx5QkFBQSxDQUEwQixXQUFXLEdBQUcsQ0FBQSxHQUN4Qyx5QkFDQSxHQUFBLENBQUEsQ0FBQSxFQUFJLHlCQUF5QixDQUFBLENBQUE7QUFDakMsRUFBTyxPQUFBLHFCQUFBO0FBQ1Q7Ozs7In0=
@@ -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"}