vite-plugin-react-server 1.4.1 → 1.4.3

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 (146) hide show
  1. package/README.md +48 -313
  2. package/dist/package.json +123 -13
  3. package/dist/plugin/bundle/deferredStaticGeneration.js +14 -39
  4. package/dist/plugin/bundle/manifests.js +30 -48
  5. package/dist/plugin/config/autoDiscover/resolveAutoDiscover.d.ts.map +1 -1
  6. package/dist/plugin/config/autoDiscover/resolveAutoDiscover.js +4 -1
  7. package/dist/plugin/config/envPrefixFromConfig.js +12 -7
  8. package/dist/plugin/config/getCondition.d.ts.map +1 -1
  9. package/dist/plugin/config/getCondition.js +7 -5
  10. package/dist/plugin/dev-server/configureReactServer.server.d.ts.map +1 -1
  11. package/dist/plugin/dev-server/configureReactServer.server.js +5 -8
  12. package/dist/plugin/dev-server/plugin.client.d.ts.map +1 -1
  13. package/dist/plugin/dev-server/plugin.client.js +2 -1
  14. package/dist/plugin/dev-server/plugin.server.d.ts.map +1 -1
  15. package/dist/plugin/dev-server/plugin.server.js +2 -36
  16. package/dist/plugin/dev-server/virtualRscHmrPlugin.js +23 -23
  17. package/dist/plugin/environments/createBuildEventPlugin.js +88 -98
  18. package/dist/plugin/environments/createEnvironmentPlugin.js +222 -250
  19. package/dist/plugin/error/index.d.ts +1 -2
  20. package/dist/plugin/error/index.d.ts.map +1 -1
  21. package/dist/plugin/error/index.js +2 -3
  22. package/dist/plugin/error/panicThresholdHandler.js +14 -3
  23. package/dist/plugin/error/setupGlobalErrorHandler.d.ts.map +1 -1
  24. package/dist/plugin/error/setupGlobalErrorHandler.js +23 -16
  25. package/dist/plugin/helpers/createRscRenderHelpers.d.ts +0 -5
  26. package/dist/plugin/helpers/createRscRenderHelpers.d.ts.map +1 -1
  27. package/dist/plugin/helpers/createRscRenderHelpers.js +32 -55
  28. package/dist/plugin/helpers/createSharedLoader.d.ts.map +1 -1
  29. package/dist/plugin/helpers/createSharedLoader.js +4 -2
  30. package/dist/plugin/helpers/headlessStreamReuseHandler.js +30 -22
  31. package/dist/plugin/helpers/headlessStreamState.d.ts +0 -38
  32. package/dist/plugin/helpers/headlessStreamState.d.ts.map +1 -1
  33. package/dist/plugin/helpers/headlessStreamState.js +15 -76
  34. package/dist/plugin/helpers/index.d.ts +0 -3
  35. package/dist/plugin/helpers/index.d.ts.map +1 -1
  36. package/dist/plugin/helpers/index.js +1 -4
  37. package/dist/plugin/helpers/requestInfo.d.ts.map +1 -1
  38. package/dist/plugin/helpers/requestInfo.js +3 -3
  39. package/dist/plugin/helpers/resolveComponent.d.ts.map +1 -1
  40. package/dist/plugin/helpers/resolveComponent.js +4 -2
  41. package/dist/plugin/helpers/workerCleanup.d.ts +1 -12
  42. package/dist/plugin/helpers/workerCleanup.d.ts.map +1 -1
  43. package/dist/plugin/helpers/workerCleanup.js +1 -1
  44. package/dist/plugin/index.client.d.ts +5 -0
  45. package/dist/plugin/index.client.d.ts.map +1 -0
  46. package/dist/plugin/index.client.js +4 -0
  47. package/dist/plugin/index.d.ts +4 -3
  48. package/dist/plugin/index.d.ts.map +1 -1
  49. package/dist/plugin/index.js +10 -5
  50. package/dist/plugin/index.server.d.ts +5 -0
  51. package/dist/plugin/index.server.d.ts.map +1 -0
  52. package/dist/plugin/index.server.js +4 -0
  53. package/dist/plugin/loader/directives/index.d.ts +0 -1
  54. package/dist/plugin/loader/directives/index.d.ts.map +1 -1
  55. package/dist/plugin/loader/directives/index.js +1 -2
  56. package/dist/plugin/metrics/createWorkerStartupMetrics.js +31 -13
  57. package/dist/plugin/orchestrator/createPluginOrchestrator.client.js +41 -38
  58. package/dist/plugin/orchestrator/createPluginOrchestrator.server.js +43 -46
  59. package/dist/plugin/plugin.client.js +2 -2
  60. package/dist/plugin/plugin.server.js +2 -2
  61. package/dist/plugin/react-static/createBuildLoader.client.js +12 -6
  62. package/dist/plugin/react-static/createBuildLoader.server.js +255 -235
  63. package/dist/plugin/react-static/plugin.client.js +684 -770
  64. package/dist/plugin/react-static/plugin.server.js +517 -603
  65. package/dist/plugin/react-static/processCssFilesForPages.js +103 -88
  66. package/dist/plugin/react-static/renderPage.client.js +455 -529
  67. package/dist/plugin/react-static/renderPage.server.js +485 -508
  68. package/dist/plugin/react-static/renderPagesBatched.js +277 -275
  69. package/dist/plugin/react-static/rscToHtmlStream.client.js +48 -29
  70. package/dist/plugin/react-static/rscToHtmlStream.server.js +62 -37
  71. package/dist/plugin/react-static/temporaryReferences.server.js +11 -2
  72. package/dist/plugin/stream/createMainThreadHandlers.js +40 -31
  73. package/dist/plugin/stream/renderRscStream.server.d.ts.map +1 -1
  74. package/dist/plugin/stream/renderRscStream.server.js +127 -144
  75. package/dist/plugin/transformer/createTransformerPlugin.js +226 -265
  76. package/dist/plugin/utils/checkReactVersion.d.ts +7 -0
  77. package/dist/plugin/utils/checkReactVersion.d.ts.map +1 -0
  78. package/dist/plugin/utils/checkReactVersion.js +23 -0
  79. package/dist/plugin/utils/envUrls.node.js +12 -11
  80. package/dist/plugin/vendor/vendor-alias.js +84 -114
  81. package/dist/plugin/vendor/vendor.client.d.ts.map +1 -1
  82. package/dist/plugin/vendor/vendor.client.js +1 -3
  83. package/dist/plugin/worker/rsc/handleRscRender.d.ts.map +1 -1
  84. package/dist/plugin/worker/rsc/handleRscRender.js +3 -1
  85. package/dist/tsconfig.tsbuildinfo +1 -1
  86. package/package.json +123 -13
  87. package/plugin/config/autoDiscover/resolveAutoDiscover.ts +4 -0
  88. package/plugin/config/getCondition.ts +6 -4
  89. package/plugin/dev-server/configureReactServer.server.ts +7 -10
  90. package/plugin/dev-server/plugin.client.ts +2 -0
  91. package/plugin/dev-server/plugin.server.ts +2 -49
  92. package/plugin/error/index.ts +1 -2
  93. package/plugin/error/setupGlobalErrorHandler.ts +24 -25
  94. package/plugin/helpers/createRscRenderHelpers.ts +0 -29
  95. package/plugin/helpers/createSharedLoader.ts +6 -1
  96. package/plugin/helpers/headlessStreamState.ts +0 -69
  97. package/plugin/helpers/index.ts +0 -3
  98. package/plugin/helpers/requestInfo.ts +1 -2
  99. package/plugin/helpers/resolveComponent.ts +6 -1
  100. package/plugin/helpers/workerCleanup.ts +1 -38
  101. package/plugin/index.client.ts +4 -0
  102. package/plugin/index.server.ts +4 -0
  103. package/plugin/index.ts +12 -5
  104. package/plugin/loader/directives/index.ts +0 -1
  105. package/plugin/plugin.client.ts +1 -1
  106. package/plugin/plugin.server.ts +1 -1
  107. package/plugin/stream/renderRscStream.server.ts +3 -0
  108. package/plugin/transformer/README.md +1 -1
  109. package/plugin/utils/checkReactVersion.ts +28 -0
  110. package/plugin/vendor/vendor.client.ts +0 -2
  111. package/plugin/worker/html/README.md +1 -1
  112. package/plugin/worker/rsc/README.md +1 -1
  113. package/plugin/worker/rsc/handleRscRender.ts +2 -0
  114. package/scripts/generate-toc.mjs +27 -294
  115. package/dist/plugin/error/assertPanic.d.ts +0 -2
  116. package/dist/plugin/error/assertPanic.d.ts.map +0 -1
  117. package/dist/plugin/error/assertPanic.js +0 -15
  118. package/dist/plugin/error/directiveError.d.ts +0 -13
  119. package/dist/plugin/error/directiveError.d.ts.map +0 -1
  120. package/dist/plugin/error/directiveError.js +0 -21
  121. package/dist/plugin/error/enhanceError.d.ts +0 -14
  122. package/dist/plugin/error/enhanceError.d.ts.map +0 -1
  123. package/dist/plugin/error/enhanceError.js +0 -24
  124. package/dist/plugin/helpers/createSafePageComponent.d.ts +0 -36
  125. package/dist/plugin/helpers/createSafePageComponent.d.ts.map +0 -1
  126. package/dist/plugin/helpers/createSafePageComponent.js +0 -50
  127. package/dist/plugin/helpers/moduleResolver.d.ts +0 -25
  128. package/dist/plugin/helpers/moduleResolver.d.ts.map +0 -1
  129. package/dist/plugin/helpers/moduleResolver.js +0 -64
  130. package/dist/plugin/helpers/stashReturnValue.d.ts +0 -3
  131. package/dist/plugin/helpers/stashReturnValue.d.ts.map +0 -1
  132. package/dist/plugin/helpers/stashReturnValue.js +0 -23
  133. package/dist/plugin/helpers/workerManager.d.ts +0 -5
  134. package/dist/plugin/helpers/workerManager.d.ts.map +0 -1
  135. package/dist/plugin/helpers/workerManager.js +0 -18
  136. package/dist/plugin/loader/directives/collectExportsFromModule.d.ts +0 -6
  137. package/dist/plugin/loader/directives/collectExportsFromModule.d.ts.map +0 -1
  138. package/dist/plugin/loader/directives/collectExportsFromModule.js +0 -24
  139. package/plugin/error/assertPanic.ts +0 -9
  140. package/plugin/error/directiveError.ts +0 -29
  141. package/plugin/error/enhanceError.ts +0 -41
  142. package/plugin/helpers/createSafePageComponent.ts +0 -64
  143. package/plugin/helpers/moduleResolver.ts +0 -91
  144. package/plugin/helpers/stashReturnValue.ts +0 -19
  145. package/plugin/helpers/workerManager.ts +0 -16
  146. package/plugin/loader/directives/collectExportsFromModule.ts +0 -25
@@ -1,269 +1,230 @@
1
- import { perEnvironmentState } from "vite";
2
- import { createTransformer } from "../loader/createTransformer.js";
3
- import { resolveOptions } from "../config/resolveOptions.js";
4
- import { readFileSync } from "node:fs";
5
- import { resolve, join } from "node:path";
6
- import { getNodeEnv, isValidEnv } from "../config/getNodeEnv.js";
7
- // import { getEnvironmentName } from "../env/plugin.js";
8
- import { DEFAULT_CONFIG } from "../config/defaults.js";
9
- import { resolveRegExp } from "../config/resolveRegExp.js";
10
- import { userProjectRoot } from "../root.js";
11
- import { createDefaultModuleID } from "../config/createModuleID.js";
12
- export const createTransformerPlugin = (options) => {
13
- return (userOptions) => {
14
- const { name } = options;
15
- // CRITICAL: Use per-environment state to prevent cross-environment cache contamination
16
- // This fixes the issue where server environment cached modules affect client environment builds
17
- const transformationCache = perEnvironmentState(() => new Map());
18
- const defaultEnvironment = options.defaultEnvironment ?? (name === "client" ? "client" : "server");
19
- const allowedEnvironments = options.allowedEnvironments ??
20
- (name === "client"
21
- ? defaultEnvironment === "client"
22
- ? ["client", "ssr"]
23
- : ["client"]
24
- : defaultEnvironment === "server"
25
- ? ["server", "ssr"]
26
- : ["server"]);
27
- const logPrefix = `[vite-plugin-react-server:transform-${defaultEnvironment}-as-${name}]`;
28
- const resolvedOptionsResult = resolveOptions(userOptions);
29
- if (resolvedOptionsResult.type === "error")
30
- throw resolvedOptionsResult.error;
31
- const { userOptions: resolvedUserOptions } = resolvedOptionsResult;
32
- let isBuild = true;
33
- let isSSR = true;
34
- const nodeEnv = getNodeEnv(process.env.NODE_ENV);
35
- let mode = nodeEnv;
36
- let runtimeResolvedUserOptions = resolvedUserOptions;
37
- // Use global cache for transformation results to ensure consistent hashing across all plugin instances
38
- const outDir = resolvedUserOptions.build.outDir || "dist";
39
- const serverDir = join(outDir, resolvedUserOptions.build.server || "server");
40
- const clientDir = join(outDir, resolvedUserOptions.build.client || "client");
41
- const staticDir = join(outDir, resolvedUserOptions.build.static || "static");
42
- const modulePattern = resolveRegExp(userOptions.autoDiscover?.modulePattern ??
43
- DEFAULT_CONFIG.AUTO_DISCOVER.modulePattern);
44
- const nodeModulesPattern = resolveRegExp(userOptions.autoDiscover?.vendorPattern ??
45
- DEFAULT_CONFIG.AUTO_DISCOVER.vendorPattern);
46
- const noDist = (id) => {
47
- // Allow files from test fixtures and project root
48
- if (id.startsWith(userProjectRoot) ||
49
- id.startsWith(join(userProjectRoot, outDir)) ||
50
- id.startsWith(join(outDir, staticDir)) ||
51
- id.startsWith(join(outDir, serverDir)) ||
52
- id.startsWith(join(outDir, clientDir))) {
53
- return true;
1
+ /**
2
+ * vite-plugin-react-server
3
+ * Copyright (c) Nico Brinkkemper
4
+ * MIT License
5
+ */
6
+ import { perEnvironmentState } from 'vite';
7
+ import { createTransformer } from '../loader/createTransformer.js';
8
+ import { resolveOptions } from '../config/resolveOptions.js';
9
+ import { readFileSync } from 'node:fs';
10
+ import { join, resolve } from 'node:path';
11
+ import { getNodeEnv, isValidEnv } from '../config/getNodeEnv.js';
12
+ import { DEFAULT_CONFIG } from '../config/defaults.js';
13
+ import { resolveRegExp } from '../config/resolveRegExp.js';
14
+ import { userProjectRoot } from '../root.js';
15
+ import { createDefaultModuleID } from '../config/createModuleID.js';
16
+
17
+ const createTransformerPlugin = (options) => {
18
+ return (userOptions) => {
19
+ const { name } = options;
20
+ const transformationCache = perEnvironmentState(() => /* @__PURE__ */ new Map());
21
+ const defaultEnvironment = options.defaultEnvironment ?? (name === "client" ? "client" : "server");
22
+ const allowedEnvironments = options.allowedEnvironments ?? (name === "client" ? defaultEnvironment === "client" ? ["client", "ssr"] : ["client"] : defaultEnvironment === "server" ? ["server", "ssr"] : ["server"]);
23
+ const logPrefix = `[vite-plugin-react-server:transform-${defaultEnvironment}-as-${name}]`;
24
+ const resolvedOptionsResult = resolveOptions(userOptions);
25
+ if (resolvedOptionsResult.type === "error")
26
+ throw resolvedOptionsResult.error;
27
+ const { userOptions: resolvedUserOptions } = resolvedOptionsResult;
28
+ let isBuild = true;
29
+ let isSSR = true;
30
+ const nodeEnv = getNodeEnv(process.env.NODE_ENV);
31
+ let mode = nodeEnv;
32
+ let runtimeResolvedUserOptions = resolvedUserOptions;
33
+ const outDir = resolvedUserOptions.build.outDir || "dist";
34
+ const serverDir = join(
35
+ outDir,
36
+ resolvedUserOptions.build.server || "server"
37
+ );
38
+ const clientDir = join(
39
+ outDir,
40
+ resolvedUserOptions.build.client || "client"
41
+ );
42
+ const staticDir = join(
43
+ outDir,
44
+ resolvedUserOptions.build.static || "static"
45
+ );
46
+ const modulePattern = resolveRegExp(
47
+ userOptions.autoDiscover?.modulePattern ?? DEFAULT_CONFIG.AUTO_DISCOVER.modulePattern
48
+ );
49
+ const nodeModulesPattern = resolveRegExp(
50
+ userOptions.autoDiscover?.vendorPattern ?? DEFAULT_CONFIG.AUTO_DISCOVER.vendorPattern
51
+ );
52
+ const noDist = (id) => {
53
+ if (id.startsWith(userProjectRoot) || id.startsWith(join(userProjectRoot, outDir)) || id.startsWith(join(outDir, staticDir)) || id.startsWith(join(outDir, serverDir)) || id.startsWith(join(outDir, clientDir))) {
54
+ return true;
55
+ }
56
+ return false;
57
+ };
58
+ return {
59
+ name: `vite-plugin-react-server:transform-${name}`,
60
+ enforce: "post",
61
+ // CRITICAL: Enable per-environment hooks during dev to prevent cache contamination
62
+ perEnvironmentStartEndDuringDev: true,
63
+ // Note: Removed applyToEnvironment - let transform hook handle filtering
64
+ // With --app builds, applyToEnvironment may not be called correctly
65
+ configResolved(config) {
66
+ isBuild = config.command === "build";
67
+ isSSR = Boolean(config.build.ssr);
68
+ mode = config.mode;
69
+ if (!isValidEnv(mode)) {
70
+ throw new Error(`Invalid mode: ${mode}`);
71
+ }
72
+ const runtimeOptionsResult = resolveOptions({
73
+ ...userOptions,
74
+ loader: {
75
+ ...userOptions.loader,
76
+ mode
77
+ }
78
+ }, true);
79
+ if (runtimeOptionsResult.type === "success") {
80
+ runtimeResolvedUserOptions = runtimeOptionsResult.userOptions;
81
+ }
82
+ if (runtimeResolvedUserOptions.loader) {
83
+ runtimeResolvedUserOptions.loader.moduleID = createDefaultModuleID(
84
+ runtimeResolvedUserOptions,
85
+ {
86
+ command: config.command,
87
+ mode: config.mode},
88
+ mode
89
+ );
90
+ }
91
+ const logger = config.customLogger || config.logger;
92
+ if (runtimeResolvedUserOptions.verbose) {
93
+ logger.info(
94
+ `${logPrefix} configResolved: isBuild=${isBuild} isSSR=${isSSR} mode=${mode} allowed=${JSON.stringify(
95
+ allowedEnvironments
96
+ )} defaultEnv=${defaultEnvironment} importServerPath=${runtimeResolvedUserOptions.loader?.importServerPath}`
97
+ );
98
+ }
99
+ },
100
+ async buildStart() {
101
+ },
102
+ transform: {
103
+ order: "post",
104
+ // when transforming to:
105
+ // dist/server / env=server - it adds registerClientReference and registerServerReference based on directive (ssg portable)
106
+ // dist/client / env=ssr - removes use client directive and hides server modules, hides client entry or without exports (ssg portable)
107
+ // dist/static / env=client - removes use client directive and hides server modules, emits client entry (and is browser portable)
108
+ async handler(code, id, { ssr } = {}) {
109
+ if (nodeModulesPattern.test(id) || !modulePattern.test(id) || !noDist(id)) {
110
+ return null;
111
+ }
112
+ let [, normalizedPath] = resolvedUserOptions.normalizer(id);
113
+ const normalizedId = id.replace(/\\/g, "/");
114
+ const normalizedServerDir = serverDir.replace(/\\/g, "/");
115
+ const normalizedClientDir = clientDir.replace(/\\/g, "/");
116
+ const isFromServerBuild = normalizedId.includes(`/${normalizedServerDir}/`) || normalizedId.includes(`dist/server/`);
117
+ const isFromClientBuild = normalizedId.includes(`/${normalizedClientDir}/`) || normalizedId.includes(`dist/client/`);
118
+ const isFromStaticBuild = normalizedId.includes(`dist/static/`);
119
+ const isBuiltFile = isBuild && /-[a-zA-Z0-9_]{6,}\.(js|mjs|cjs)$/.test(normalizedId);
120
+ const isAlreadyTransformed = code.includes(
121
+ runtimeResolvedUserOptions.loader?.registerClientReferenceName ?? "registerClientReference"
122
+ );
123
+ if (isAlreadyTransformed) {
124
+ if (runtimeResolvedUserOptions.verbose) {
125
+ this.environment?.logger?.info(
126
+ `[react-${name}-transform] Encountered already transformed file: ${id}. This indicates two transformers are running on the same file: ${this.environment?.name} and ${Object.entries(this.environment?.plugins ?? {}).map(([name2, plugin]) => `${name2} (${plugin.name})`).join(", ")}`
127
+ );
128
+ this.environment?.logger?.info("");
54
129
  }
55
- return false;
56
- };
57
- return {
58
- name: `vite-plugin-react-server:transform-${name}`,
59
- enforce: "post",
60
- // CRITICAL: Enable per-environment hooks during dev to prevent cache contamination
61
- perEnvironmentStartEndDuringDev: true,
62
- // Note: Removed applyToEnvironment - let transform hook handle filtering
63
- // With --app builds, applyToEnvironment may not be called correctly
64
- configResolved(config) {
65
- isBuild = config.command === "build";
66
- isSSR = Boolean(config.build.ssr);
67
- mode = config.mode;
68
- if (!isValidEnv(mode)) {
69
- throw new Error(`Invalid mode: ${mode}`);
70
- }
71
- // CRITICAL: Re-resolve options with runtime mode to get correct importServerPath
72
- // This ensures test mode uses react-server-dom-esm/server.node instead of server
73
- // Force re-resolve to avoid cached moduleID functions from different build contexts
74
- const runtimeOptionsResult = resolveOptions({
75
- ...userOptions,
76
- loader: {
77
- ...userOptions.loader,
78
- mode: mode,
79
- },
80
- }, true); // Force resolve to bypass cache
81
- if (runtimeOptionsResult.type === "success") {
82
- runtimeResolvedUserOptions = runtimeOptionsResult.userOptions;
83
- }
84
- // CRITICAL: Update moduleID function with correct configEnv for build mode
85
- // This ensures client component hashing uses the correct build context
86
- // ALWAYS recreate the moduleID to ensure it matches the current command
87
- if (runtimeResolvedUserOptions.loader) {
88
- runtimeResolvedUserOptions.loader.moduleID = createDefaultModuleID(runtimeResolvedUserOptions, {
89
- command: config.command,
90
- mode: config.mode,
91
- isSsrBuild: isSSR,
92
- isPreview: false,
93
- }, mode);
94
- }
95
- // Note: condition override is set in env plugin during config phase
96
- // Verbose summary (config hook has void context, use config logger)
97
- const logger = config.customLogger || config.logger;
98
- // Only log in verbose mode
99
- if (runtimeResolvedUserOptions.verbose) {
100
- logger.info(`${logPrefix} configResolved: isBuild=${isBuild} isSSR=${isSSR} mode=${mode} allowed=${JSON.stringify(allowedEnvironments)} defaultEnv=${defaultEnvironment} importServerPath=${runtimeResolvedUserOptions.loader?.importServerPath}`);
101
- }
102
- },
103
- async buildStart() {
104
- // No longer load static manifest - rely on hash coordination to ensure consistent hashes
105
- // This removes the file I/O dependency and allows parallel builds
106
- },
107
- transform: {
108
- order: "post",
109
- // when transforming to:
110
- // dist/server / env=server - it adds registerClientReference and registerServerReference based on directive (ssg portable)
111
- // dist/client / env=ssr - removes use client directive and hides server modules, hides client entry or without exports (ssg portable)
112
- // dist/static / env=client - removes use client directive and hides server modules, emits client entry (and is browser portable)
113
- async handler(code, id, { ssr } = {}) {
114
- if (nodeModulesPattern.test(id) ||
115
- !modulePattern.test(id) ||
116
- !noDist(id)) {
117
- return null;
118
- }
119
- let [, normalizedPath] = resolvedUserOptions.normalizer(id);
120
- // Check if this is a built file that doesn't need transformation
121
- // Normalize paths to handle cross-platform differences
122
- const normalizedId = id.replace(/\\/g, "/");
123
- const normalizedServerDir = serverDir.replace(/\\/g, "/");
124
- const normalizedClientDir = clientDir.replace(/\\/g, "/");
125
- // Check if the file is from a build output directory
126
- const isFromServerBuild = normalizedId.includes(`/${normalizedServerDir}/`) ||
127
- normalizedId.includes(`dist/server/`);
128
- const isFromClientBuild = normalizedId.includes(`/${normalizedClientDir}/`) ||
129
- normalizedId.includes(`dist/client/`);
130
- const isFromStaticBuild = normalizedId.includes(`dist/static/`);
131
- // Check if this looks like a built/hashed file (should never be transformed)
132
- // Built files have hashes and are already processed
133
- const isBuiltFile = isBuild && /-[a-zA-Z0-9_]{6,}\.(js|mjs|cjs)$/.test(normalizedId);
134
- // Check if this file is already transformed (contains registerClientReference)
135
- const isAlreadyTransformed = code.includes(runtimeResolvedUserOptions.loader?.registerClientReferenceName ??
136
- "registerClientReference");
137
- if (isAlreadyTransformed) {
138
- if (runtimeResolvedUserOptions.verbose) {
139
- this.environment?.logger?.info(`[react-${name}-transform] Encountered already transformed file: ${id}. This indicates two transformers are running on the same file: ${this.environment?.name} and ${Object.entries(this.environment?.plugins ?? {})
140
- .map(([name, plugin]) => `${name} (${plugin.name})`)
141
- .join(", ")}`);
142
- this.environment?.logger?.info('');
143
- }
144
- return {
145
- code: code,
146
- map: null,
147
- };
148
- }
149
- // Check if we've already transformed this module to avoid double-hashing
150
- // Include environment context in cache key since different environments need different transformations
151
- const isServerEnv = this.environment?.name === "server";
152
- // CRITICAL: Use per-environment cache to prevent cross-environment contamination
153
- const envCache = transformationCache(this);
154
- const cacheKey = `${normalizedPath}:${isServerEnv ? "server" : "client"}:${code}`;
155
- if (envCache.has(cacheKey)) {
156
- if (runtimeResolvedUserOptions.verbose) {
157
- this.environment?.logger?.info(`[react-${name}-transform] Using cached transformation for: ${normalizedPath} (${isServerEnv ? "server" : "client"}) env=${this.environment?.name}`);
158
- }
159
- return envCache.get(cacheKey);
160
- }
161
- // Get the original source content for consistent hashing
162
- // Read the file directly to ensure we use the original content, not transformed code
163
- let originalSourceContent;
164
- try {
165
- const sourcePath = resolve(userProjectRoot, id);
166
- originalSourceContent = readFileSync(sourcePath, "utf-8");
167
- }
168
- catch (error) {
169
- // Fallback to the provided code if we can't read the file
170
- originalSourceContent = code;
171
- }
172
- // Use the original normalized path for moduleID function calls
173
- // This ensures registerClientReference calls use the correct paths
174
- let finalModuleID = runtimeResolvedUserOptions.loader?.moduleID
175
- ? runtimeResolvedUserOptions.loader.moduleID(normalizedPath, originalSourceContent)
176
- : normalizedPath;
177
- if (runtimeResolvedUserOptions.verbose) {
178
- this.environment?.logger?.info(`[react-${name}-transform] ModuleID transformation: ${normalizedPath} -> ${finalModuleID}`);
179
- }
180
- // Determine if this is a server environment
181
- // Check both the environment name and if we're doing server-side rendering for static generation
182
- const envName = this.environment?.name?.toLowerCase() || "";
183
- const isServerEnvironment = envName === "server" || envName === "rsc" || envName === "react-server";
184
- const transformer = createTransformer({
185
- parseFn: (source) => {
186
- const ast = this.parse(source, {
187
- allowReturnOutsideFunction: true,
188
- jsx: true,
189
- });
190
- return ast;
191
- },
192
- options: {
193
- loader: runtimeResolvedUserOptions.loader,
194
- verbose: runtimeResolvedUserOptions.verbose,
195
- panicThreshold: runtimeResolvedUserOptions.panicThreshold,
196
- logger: this.environment?.logger,
197
- moduleBase: userOptions.moduleBase ?? "",
198
- },
199
- // Pass the actual environment context to the transformer
200
- // Only the actual "server" environment should transform client components to registerClientReference
201
- // SSR environment needs actual React components, not placeholders
202
- isServerEnvironment: isServerEnvironment,
203
- ssr: ssr,
204
- });
205
- // Skip files from output directories that are already built and transformed
206
- // But allow transformation of server-built client components that need registerClientReference
207
- if (isFromServerBuild ||
208
- isFromClientBuild ||
209
- isFromStaticBuild ||
210
- isBuiltFile) {
211
- const buildType = isFromServerBuild
212
- ? "server"
213
- : isFromClientBuild
214
- ? "client"
215
- : isFromStaticBuild
216
- ? "static"
217
- : "built";
218
- // Allow transformation of server-built client components
219
- if (isFromServerBuild &&
220
- runtimeResolvedUserOptions.loader?.isClientComponentByName?.(id)) {
221
- if (runtimeResolvedUserOptions.verbose) {
222
- this.environment?.logger?.info(`[react-${name}-transform] Allowing transformation of server-built client component: ${id}`);
223
- }
224
- // Don't skip - let it fall through to transformer
225
- }
226
- else {
227
- if (runtimeResolvedUserOptions.verbose) {
228
- this.environment?.logger?.info(`[react-${name}-transform] Skipping built file from ${buildType} build: ${id}`);
229
- }
230
- return {
231
- code: code,
232
- map: null,
233
- };
234
- }
235
- }
236
- const transformResult = await transformer(code, normalizedPath, finalModuleID);
237
- // If transformer returns null (e.g., for built files), return original code
238
- if (!transformResult) {
239
- return { code, map: null };
240
- }
241
- const { code: transformed, map } = transformResult;
242
- // Store the transformation result in per-environment cache
243
- const result = { code: transformed, map };
244
- envCache.set(cacheKey, result);
245
- // Logging for verbose mode
246
- if (runtimeResolvedUserOptions.verbose) {
247
- const hasDirectives = code.includes('"use client"') ||
248
- code.includes('"use server"') ||
249
- code.includes("'use client'") ||
250
- code.includes("'use server'");
251
- if (transformed !== code) {
252
- this.environment?.logger?.info(`[react-${name}-transform] ` +
253
- id.split("/").pop() +
254
- (code.startsWith('"use client"') ? " (client)" : "") +
255
- (hasDirectives ? " (directives processed)" : ""));
256
- this.environment?.logger?.info(`[react-${name}-transform] ` + transformed.slice(0, 100) + "...");
257
- }
258
- else if (hasDirectives) {
259
- this.environment?.logger?.info(`[react-${name}-transform] ` +
260
- id.split("/").pop() +
261
- " (directives already processed)");
262
- }
263
- }
264
- return result;
265
- },
130
+ return {
131
+ code,
132
+ map: null
133
+ };
134
+ }
135
+ const isServerEnv = this.environment?.name === "server";
136
+ const envCache = transformationCache(this);
137
+ const cacheKey = `${normalizedPath}:${isServerEnv ? "server" : "client"}:${code}`;
138
+ if (envCache.has(cacheKey)) {
139
+ if (runtimeResolvedUserOptions.verbose) {
140
+ this.environment?.logger?.info(
141
+ `[react-${name}-transform] Using cached transformation for: ${normalizedPath} (${isServerEnv ? "server" : "client"}) env=${this.environment?.name}`
142
+ );
143
+ }
144
+ return envCache.get(cacheKey);
145
+ }
146
+ let originalSourceContent;
147
+ try {
148
+ const sourcePath = resolve(userProjectRoot, id);
149
+ originalSourceContent = readFileSync(sourcePath, "utf-8");
150
+ } catch (error) {
151
+ originalSourceContent = code;
152
+ }
153
+ let finalModuleID = runtimeResolvedUserOptions.loader?.moduleID ? runtimeResolvedUserOptions.loader.moduleID(
154
+ normalizedPath,
155
+ originalSourceContent
156
+ ) : normalizedPath;
157
+ if (runtimeResolvedUserOptions.verbose) {
158
+ this.environment?.logger?.info(
159
+ `[react-${name}-transform] ModuleID transformation: ${normalizedPath} -> ${finalModuleID}`
160
+ );
161
+ }
162
+ const envName = this.environment?.name?.toLowerCase() || "";
163
+ const isServerEnvironment = envName === "server" || envName === "rsc" || envName === "react-server";
164
+ const transformer = createTransformer({
165
+ options: {
166
+ loader: runtimeResolvedUserOptions.loader,
167
+ verbose: runtimeResolvedUserOptions.verbose,
168
+ panicThreshold: runtimeResolvedUserOptions.panicThreshold,
169
+ logger: this.environment?.logger,
170
+ moduleBase: userOptions.moduleBase ?? ""
266
171
  },
267
- };
172
+ // Pass the actual environment context to the transformer
173
+ // Only the actual "server" environment should transform client components to registerClientReference
174
+ // SSR environment needs actual React components, not placeholders
175
+ isServerEnvironment});
176
+ if (isFromServerBuild || isFromClientBuild || isFromStaticBuild || isBuiltFile) {
177
+ const buildType = isFromServerBuild ? "server" : isFromClientBuild ? "client" : isFromStaticBuild ? "static" : "built";
178
+ if (isFromServerBuild && runtimeResolvedUserOptions.loader?.isClientComponentByName?.(id)) {
179
+ if (runtimeResolvedUserOptions.verbose) {
180
+ this.environment?.logger?.info(
181
+ `[react-${name}-transform] Allowing transformation of server-built client component: ${id}`
182
+ );
183
+ }
184
+ } else {
185
+ if (runtimeResolvedUserOptions.verbose) {
186
+ this.environment?.logger?.info(
187
+ `[react-${name}-transform] Skipping built file from ${buildType} build: ${id}`
188
+ );
189
+ }
190
+ return {
191
+ code,
192
+ map: null
193
+ };
194
+ }
195
+ }
196
+ const transformResult = await transformer(
197
+ code,
198
+ normalizedPath,
199
+ finalModuleID
200
+ );
201
+ if (!transformResult) {
202
+ return { code, map: null };
203
+ }
204
+ const { code: transformed, map } = transformResult;
205
+ const result = { code: transformed, map };
206
+ envCache.set(cacheKey, result);
207
+ if (runtimeResolvedUserOptions.verbose) {
208
+ const hasDirectives = code.includes('"use client"') || code.includes('"use server"') || code.includes("'use client'") || code.includes("'use server'");
209
+ if (transformed !== code) {
210
+ this.environment?.logger?.info(
211
+ `[react-${name}-transform] ` + id.split("/").pop() + (code.startsWith('"use client"') ? " (client)" : "") + (hasDirectives ? " (directives processed)" : "")
212
+ );
213
+ this.environment?.logger?.info(
214
+ `[react-${name}-transform] ` + transformed.slice(0, 100) + "..."
215
+ );
216
+ } else if (hasDirectives) {
217
+ this.environment?.logger?.info(
218
+ `[react-${name}-transform] ` + id.split("/").pop() + " (directives already processed)"
219
+ );
220
+ }
221
+ }
222
+ return result;
223
+ }
224
+ }
268
225
  };
226
+ };
269
227
  };
228
+
229
+ export { createTransformerPlugin };
230
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"createTransformerPlugin.js","sources":["../../../plugin/transformer/createTransformerPlugin.ts"],"sourcesContent":["import type { Plugin } from \"vite\";\nimport { perEnvironmentState } from \"vite\";\nimport type { VitePluginFn } from \"../types.js\";\nimport { createTransformer } from \"../loader/createTransformer.js\";\nimport type { Program } from \"acorn\";\nimport { resolveOptions } from \"../config/resolveOptions.js\";\nimport { readFileSync } from \"node:fs\";\nimport { resolve, join } from \"node:path\";\nimport { getNodeEnv, isValidEnv } from \"../config/getNodeEnv.js\";\n\n// import { getEnvironmentName } from \"../env/plugin.js\";\nimport { DEFAULT_CONFIG } from \"../config/defaults.js\";\nimport { resolveRegExp } from \"../config/resolveRegExp.js\";\nimport { userProjectRoot } from \"../root.js\";\nimport { createDefaultModuleID } from \"../config/createModuleID.js\";\n\nexport interface TransformerPluginOptions {\n  name: string;\n  /**\n   * Optional. If omitted, sensible defaults are applied based on `name`:\n   * - name === \"client\" -> [\"client\", \"ssr\"]\n   * - name === \"server\" -> [\"server\"]\n   */\n  allowedEnvironments?: (\"client\" | \"server\" | \"ssr\")[];\n  /**\n   * Optional. If omitted, sensible defaults are applied based on `name`:\n   * - name === \"client\" -> \"client\"\n   * - name === \"server\" -> \"server\"\n   */\n  defaultEnvironment?: \"client\" | \"server\" | \"ssr\";\n}\n\nexport const createTransformerPlugin = (\n  options: TransformerPluginOptions\n): VitePluginFn => {\n  return (userOptions) => {\n    const { name } = options;\n\n    // CRITICAL: Use per-environment state to prevent cross-environment cache contamination\n    // This fixes the issue where server environment cached modules affect client environment builds\n    const transformationCache = perEnvironmentState<\n      Map<string, { code: string; map: any }>\n    >(() => new Map());\n    const defaultEnvironment =\n      options.defaultEnvironment ?? (name === \"client\" ? \"client\" : \"server\");\n    const allowedEnvironments =\n      options.allowedEnvironments ??\n      (name === \"client\"\n        ? defaultEnvironment === \"client\"\n          ? [\"client\", \"ssr\"]\n          : [\"client\"]\n        : defaultEnvironment === \"server\"\n        ? [\"server\", \"ssr\"]\n        : [\"server\"]);\n    const logPrefix = `[vite-plugin-react-server:transform-${defaultEnvironment}-as-${name}]`;\n\n    const resolvedOptionsResult = resolveOptions(userOptions);\n    if (resolvedOptionsResult.type === \"error\")\n      throw resolvedOptionsResult.error;\n    const { userOptions: resolvedUserOptions } = resolvedOptionsResult;\n\n    let isBuild = true;\n    let isSSR = true;\n    const nodeEnv = getNodeEnv(process.env.NODE_ENV);\n    let mode = nodeEnv;\n    let runtimeResolvedUserOptions = resolvedUserOptions;\n\n    // Use global cache for transformation results to ensure consistent hashing across all plugin instances\n    const outDir = resolvedUserOptions.build.outDir || \"dist\";\n    const serverDir = join(\n      outDir,\n      resolvedUserOptions.build.server || \"server\"\n    );\n    const clientDir = join(\n      outDir,\n      resolvedUserOptions.build.client || \"client\"\n    );\n    const staticDir = join(\n      outDir,\n      resolvedUserOptions.build.static || \"static\"\n    );\n    const modulePattern = resolveRegExp(\n      userOptions.autoDiscover?.modulePattern ??\n        DEFAULT_CONFIG.AUTO_DISCOVER.modulePattern\n    );\n    const nodeModulesPattern = resolveRegExp(\n      userOptions.autoDiscover?.vendorPattern ??\n        DEFAULT_CONFIG.AUTO_DISCOVER.vendorPattern\n    );\n    const noDist = (id: string) => {\n      // Allow files from test fixtures and project root\n      if (\n        id.startsWith(userProjectRoot) ||\n        id.startsWith(join(userProjectRoot, outDir)) ||\n        id.startsWith(join(outDir, staticDir)) ||\n        id.startsWith(join(outDir, serverDir)) ||\n        id.startsWith(join(outDir, clientDir))\n      ) {\n        return true;\n      }\n      return false;\n    };\n\n    return {\n      name: `vite-plugin-react-server:transform-${name}`,\n      enforce: \"post\",\n      // CRITICAL: Enable per-environment hooks during dev to prevent cache contamination\n      perEnvironmentStartEndDuringDev: true,\n      // Note: Removed applyToEnvironment - let transform hook handle filtering\n      // With --app builds, applyToEnvironment may not be called correctly\n      configResolved(config) {\n        isBuild = config.command === \"build\";\n        isSSR = Boolean(config.build.ssr);\n        mode = config.mode as \"development\" | \"production\" | \"test\";\n        if (!isValidEnv(mode)) {\n          throw new Error(`Invalid mode: ${mode}`);\n        }\n\n        // CRITICAL: Re-resolve options with runtime mode to get correct importServerPath\n        // This ensures test mode uses react-server-dom-esm/server.node instead of server\n        // Force re-resolve to avoid cached moduleID functions from different build contexts\n        const runtimeOptionsResult = resolveOptions({\n          ...userOptions,\n          loader: {\n            ...userOptions.loader,\n            mode: mode,\n          },\n        }, true); // Force resolve to bypass cache\n        if (runtimeOptionsResult.type === \"success\") {\n          runtimeResolvedUserOptions = runtimeOptionsResult.userOptions;\n        }\n\n        // CRITICAL: Update moduleID function with correct configEnv for build mode\n        // This ensures client component hashing uses the correct build context\n        // ALWAYS recreate the moduleID to ensure it matches the current command\n        if (runtimeResolvedUserOptions.loader) {\n          runtimeResolvedUserOptions.loader.moduleID = createDefaultModuleID(\n            runtimeResolvedUserOptions,\n            {\n              command: config.command,\n              mode: config.mode,\n              isSsrBuild: isSSR,\n              isPreview: false,\n            },\n            mode\n          );\n        }\n\n        // Note: condition override is set in env plugin during config phase\n        // Verbose summary (config hook has void context, use config logger)\n        const logger = config.customLogger || config.logger;\n        // Only log in verbose mode\n        if (runtimeResolvedUserOptions.verbose) {\n          logger.info(\n            `${logPrefix} configResolved: isBuild=${isBuild} isSSR=${isSSR} mode=${mode} allowed=${JSON.stringify(\n              allowedEnvironments\n            )} defaultEnv=${defaultEnvironment} importServerPath=${\n              runtimeResolvedUserOptions.loader?.importServerPath\n            }`\n          );\n        }\n      },\n      async buildStart() {\n        // No longer load static manifest - rely on hash coordination to ensure consistent hashes\n        // This removes the file I/O dependency and allows parallel builds\n      },\n      transform: {\n        order: \"post\",\n        // when transforming to:\n        // dist/server / env=server - it adds registerClientReference and registerServerReference based on directive (ssg portable)\n        // dist/client / env=ssr - removes use client directive and hides server modules, hides client entry or without exports (ssg portable)\n        // dist/static / env=client  -  removes use client directive and hides server modules, emits client entry (and is browser portable)\n        async handler(code, id, { ssr } = {}) {\n          if (\n            nodeModulesPattern.test(id) ||\n            !modulePattern.test(id) ||\n            !noDist(id)\n          ) {\n            return null;\n          }\n          let [, normalizedPath] = resolvedUserOptions.normalizer(id);\n\n          // Check if this is a built file that doesn't need transformation\n          // Normalize paths to handle cross-platform differences\n          const normalizedId = id.replace(/\\\\/g, \"/\");\n          const normalizedServerDir = serverDir.replace(/\\\\/g, \"/\");\n          const normalizedClientDir = clientDir.replace(/\\\\/g, \"/\");\n\n          // Check if the file is from a build output directory\n          const isFromServerBuild =\n            normalizedId.includes(`/${normalizedServerDir}/`) ||\n            normalizedId.includes(`dist/server/`);\n          const isFromClientBuild =\n            normalizedId.includes(`/${normalizedClientDir}/`) ||\n            normalizedId.includes(`dist/client/`);\n          const isFromStaticBuild = normalizedId.includes(`dist/static/`);\n\n          // Check if this looks like a built/hashed file (should never be transformed)\n          // Built files have hashes and are already processed\n          const isBuiltFile =\n            isBuild && /-[a-zA-Z0-9_]{6,}\\.(js|mjs|cjs)$/.test(normalizedId);\n\n          // Check if this file is already transformed (contains registerClientReference)\n          const isAlreadyTransformed = code.includes(\n            runtimeResolvedUserOptions.loader?.registerClientReferenceName ??\n              \"registerClientReference\"\n          );\n          if (isAlreadyTransformed) {\n            if (runtimeResolvedUserOptions.verbose) {\n              this.environment?.logger?.info(\n                `[react-${name}-transform] Encountered already transformed file: ${id}. This indicates two transformers are running on the same file: ${\n                  this.environment?.name\n                } and ${Object.entries(this.environment?.plugins ?? {})\n                  .map(([name, plugin]) => `${name} (${plugin.name})`)\n                  .join(\", \")}`\n              );\n              this.environment?.logger?.info('')\n            }\n            return {\n              code: code,\n              map: null,\n            };\n          }\n\n          // Check if we've already transformed this module to avoid double-hashing\n          // Include environment context in cache key since different environments need different transformations\n          const isServerEnv = this.environment?.name === \"server\";\n          // CRITICAL: Use per-environment cache to prevent cross-environment contamination\n          const envCache = transformationCache(this);\n          const cacheKey = `${normalizedPath}:${\n            isServerEnv ? \"server\" : \"client\"\n          }:${code}`;\n          if (envCache.has(cacheKey)) {\n            if (runtimeResolvedUserOptions.verbose) {\n              this.environment?.logger?.info(\n                `[react-${name}-transform] Using cached transformation for: ${normalizedPath} (${\n                  isServerEnv ? \"server\" : \"client\"\n                }) env=${this.environment?.name}`\n              );\n            }\n            return envCache.get(cacheKey);\n          }\n\n          // Get the original source content for consistent hashing\n          // Read the file directly to ensure we use the original content, not transformed code\n          let originalSourceContent: string;\n          try {\n            const sourcePath = resolve(userProjectRoot, id);\n            originalSourceContent = readFileSync(sourcePath, \"utf-8\");\n          } catch (error) {\n            // Fallback to the provided code if we can't read the file\n            originalSourceContent = code;\n          }\n\n          // Use the original normalized path for moduleID function calls\n          // This ensures registerClientReference calls use the correct paths\n          let finalModuleID = runtimeResolvedUserOptions.loader?.moduleID\n            ? runtimeResolvedUserOptions.loader.moduleID(\n                normalizedPath,\n                originalSourceContent\n              )\n            : normalizedPath;\n\n          if (runtimeResolvedUserOptions.verbose) {\n            this.environment?.logger?.info(\n              `[react-${name}-transform] ModuleID transformation: ${normalizedPath} -> ${finalModuleID}`\n            );\n          }\n\n          // Determine if this is a server environment\n          // Check both the environment name and if we're doing server-side rendering for static generation\n          const envName = this.environment?.name?.toLowerCase() || \"\";\n          const isServerEnvironment = envName === \"server\" || envName === \"rsc\" || envName === \"react-server\";\n\n          const transformer = createTransformer({\n            parseFn: (source) => {\n              const ast = this.parse(source, {\n                allowReturnOutsideFunction: true,\n                jsx: true,\n              }) as Program;\n              return ast;\n            },\n            options: {\n              loader: runtimeResolvedUserOptions.loader,\n              verbose: runtimeResolvedUserOptions.verbose,\n              panicThreshold: runtimeResolvedUserOptions.panicThreshold,\n              logger: this.environment?.logger,\n              moduleBase: userOptions.moduleBase ?? \"\",\n            },\n\n            // Pass the actual environment context to the transformer\n            // Only the actual \"server\" environment should transform client components to registerClientReference\n            // SSR environment needs actual React components, not placeholders\n            isServerEnvironment: isServerEnvironment,\n            ssr: ssr,\n          });\n\n          // Skip files from output directories that are already built and transformed\n          // But allow transformation of server-built client components that need registerClientReference\n\n          if (\n            isFromServerBuild ||\n            isFromClientBuild ||\n            isFromStaticBuild ||\n            isBuiltFile\n          ) {\n            const buildType = isFromServerBuild\n              ? \"server\"\n              : isFromClientBuild\n              ? \"client\"\n              : isFromStaticBuild\n              ? \"static\"\n              : \"built\";\n\n            // Allow transformation of server-built client components\n            if (\n              isFromServerBuild &&\n              runtimeResolvedUserOptions.loader?.isClientComponentByName?.(id)\n            ) {\n              if (runtimeResolvedUserOptions.verbose) {\n                this.environment?.logger?.info(\n                  `[react-${name}-transform] Allowing transformation of server-built client component: ${id}`\n                );\n              }\n              // Don't skip - let it fall through to transformer\n            } else {\n              if (runtimeResolvedUserOptions.verbose) {\n                this.environment?.logger?.info(\n                  `[react-${name}-transform] Skipping built file from ${buildType} build: ${id}`\n                );\n              }\n              return {\n                code: code,\n                map: null,\n              };\n            }\n          }\n\n          const transformResult = await transformer(\n            code,\n            normalizedPath,\n            finalModuleID\n          );\n\n          // If transformer returns null (e.g., for built files), return original code\n          if (!transformResult) {\n            return { code, map: null };\n          }\n\n          const { code: transformed, map } = transformResult;\n\n          // Store the transformation result in per-environment cache\n          const result = { code: transformed, map };\n          envCache.set(cacheKey, result);\n\n          // Logging for verbose mode\n          if (runtimeResolvedUserOptions.verbose) {\n            const hasDirectives =\n              code.includes('\"use client\"') ||\n              code.includes('\"use server\"') ||\n              code.includes(\"'use client'\") ||\n              code.includes(\"'use server'\");\n\n            if (transformed !== code) {\n              this.environment?.logger?.info(\n                `[react-${name}-transform] ` +\n                  id.split(\"/\").pop() +\n                  (code.startsWith('\"use client\"') ? \" (client)\" : \"\") +\n                  (hasDirectives ? \" (directives processed)\" : \"\")\n              );\n              this.environment?.logger?.info(\n                `[react-${name}-transform] ` + transformed.slice(0, 100) + \"...\"\n              );\n            } else if (hasDirectives) {\n              this.environment?.logger?.info(\n                `[react-${name}-transform] ` +\n                  id.split(\"/\").pop() +\n                  \" (directives already processed)\"\n              );\n            }\n          }\n\n          return result;\n        },\n      },\n    } as Plugin;\n  };\n};\n"],"names":["name"],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ /**
2
+ * Verify React experimental is available. The vendored react-server-dom-esm
3
+ * transport requires Taint APIs that only exist in React experimental builds.
4
+ * Stable React 19.x causes: "Cannot read properties of undefined (reading 'add')"
5
+ */
6
+ export declare function checkReactExperimental(): void;
7
+ //# sourceMappingURL=checkReactVersion.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"checkReactVersion.d.ts","sourceRoot":"","sources":["../../../plugin/utils/checkReactVersion.ts"],"names":[],"mappings":"AAEA;;;;GAIG;AACH,wBAAgB,sBAAsB,SAoBrC"}
@@ -0,0 +1,23 @@
1
+ /**
2
+ * vite-plugin-react-server
3
+ * Copyright (c) Nico Brinkkemper
4
+ * MIT License
5
+ */
6
+ let checked = false;
7
+ function checkReactExperimental() {
8
+ if (checked) return;
9
+ checked = true;
10
+ try {
11
+ const version = globalThis.process?.versions ? require("react").version : "";
12
+ if (version && !version.includes("experimental")) {
13
+ throw new Error(
14
+ `[vite-plugin-react-server] React ${version} is not supported. This plugin requires React experimental builds. Install with: npm install react@experimental react-dom@experimental`
15
+ );
16
+ }
17
+ } catch (e) {
18
+ if (e instanceof Error && e.message.includes("not supported")) throw e;
19
+ }
20
+ }
21
+
22
+ export { checkReactExperimental };
23
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hlY2tSZWFjdFZlcnNpb24uanMiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3BsdWdpbi91dGlscy9jaGVja1JlYWN0VmVyc2lvbi50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJsZXQgY2hlY2tlZCA9IGZhbHNlO1xuXG4vKipcbiAqIFZlcmlmeSBSZWFjdCBleHBlcmltZW50YWwgaXMgYXZhaWxhYmxlLiBUaGUgdmVuZG9yZWQgcmVhY3Qtc2VydmVyLWRvbS1lc21cbiAqIHRyYW5zcG9ydCByZXF1aXJlcyBUYWludCBBUElzIHRoYXQgb25seSBleGlzdCBpbiBSZWFjdCBleHBlcmltZW50YWwgYnVpbGRzLlxuICogU3RhYmxlIFJlYWN0IDE5LnggY2F1c2VzOiBcIkNhbm5vdCByZWFkIHByb3BlcnRpZXMgb2YgdW5kZWZpbmVkIChyZWFkaW5nICdhZGQnKVwiXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjaGVja1JlYWN0RXhwZXJpbWVudGFsKCkge1xuICBpZiAoY2hlY2tlZCkgcmV0dXJuO1xuICBjaGVja2VkID0gdHJ1ZTtcbiAgXG4gIHRyeSB7XG4gICAgLy8gVXNlIHJlcXVpcmUgZm9yIHN5bmMgYWNjZXNzIOKAlCBSZWFjdCBpcyBhbHdheXMgYXZhaWxhYmxlIGF0IHRoaXMgcG9pbnRcbiAgICBjb25zdCB2ZXJzaW9uOiBzdHJpbmcgPSBnbG9iYWxUaGlzLnByb2Nlc3M/LnZlcnNpb25zXG4gICAgICA/IHJlcXVpcmUoXCJyZWFjdFwiKS52ZXJzaW9uXG4gICAgICA6IFwiXCI7XG4gICAgXG4gICAgaWYgKHZlcnNpb24gJiYgIXZlcnNpb24uaW5jbHVkZXMoXCJleHBlcmltZW50YWxcIikpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgYFt2aXRlLXBsdWdpbi1yZWFjdC1zZXJ2ZXJdIFJlYWN0ICR7dmVyc2lvbn0gaXMgbm90IHN1cHBvcnRlZC4gYCArXG4gICAgICAgIGBUaGlzIHBsdWdpbiByZXF1aXJlcyBSZWFjdCBleHBlcmltZW50YWwgYnVpbGRzLiBgICtcbiAgICAgICAgYEluc3RhbGwgd2l0aDogbnBtIGluc3RhbGwgcmVhY3RAZXhwZXJpbWVudGFsIHJlYWN0LWRvbUBleHBlcmltZW50YWxgXG4gICAgICApO1xuICAgIH1cbiAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgIGlmIChlIGluc3RhbmNlb2YgRXJyb3IgJiYgZS5tZXNzYWdlLmluY2x1ZGVzKFwibm90IHN1cHBvcnRlZFwiKSkgdGhyb3cgZTtcbiAgfVxufVxuIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsSUFBSSxPQUFVLEdBQUEsS0FBQTtBQU9QLFNBQVMsc0JBQXlCLEdBQUE7QUFDdkMsRUFBQSxJQUFJLE9BQVMsRUFBQTtBQUNiLEVBQVUsT0FBQSxHQUFBLElBQUE7QUFFVixFQUFJLElBQUE7QUFFRixJQUFBLE1BQU0sVUFBa0IsVUFBVyxDQUFBLE9BQUEsRUFBUyxXQUN4QyxPQUFRLENBQUEsT0FBTyxFQUFFLE9BQ2pCLEdBQUEsRUFBQTtBQUVKLElBQUEsSUFBSSxPQUFXLElBQUEsQ0FBQyxPQUFRLENBQUEsUUFBQSxDQUFTLGNBQWMsQ0FBRyxFQUFBO0FBQ2hELE1BQUEsTUFBTSxJQUFJLEtBQUE7QUFBQSxRQUNSLG9DQUFvQyxPQUFPLENBQUEsc0lBQUE7QUFBQSxPQUc3QztBQUFBO0FBQ0YsV0FDTyxDQUFZLEVBQUE7QUFDbkIsSUFBQSxJQUFJLGFBQWEsS0FBUyxJQUFBLENBQUEsQ0FBRSxRQUFRLFFBQVMsQ0FBQSxlQUFlLEdBQVMsTUFBQSxDQUFBO0FBQUE7QUFFekU7Ozs7In0=