Skip to content

Commit eb33d72

Browse files
committed
feat: check consistency of JSX between swc and tsconfig
1 parent 860dba6 commit eb33d72

File tree

9 files changed

+133
-12
lines changed

9 files changed

+133
-12
lines changed

packages/core/src/check.ts

+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
import type { RsbuildConfig, RsbuildPlugin } from '@rsbuild/core';
2+
import { color } from './utils/helper';
3+
import { logger } from './utils/logger';
4+
5+
type PluginReactOptions = {
6+
compilerOptions?: Record<string, any>;
7+
};
8+
9+
const mapTsconfigJsxToSwcJsx = (swcJsx: string): string | null => {
10+
switch (swcJsx) {
11+
case 'react-jsx':
12+
case 'react-jsxdev':
13+
return 'automatic';
14+
case 'react':
15+
return 'classic';
16+
default:
17+
return 'classic';
18+
}
19+
};
20+
21+
const checkJsx = ({ compilerOptions }: PluginReactOptions): RsbuildPlugin => ({
22+
name: 'rsbuild:lib-check',
23+
setup(api) {
24+
api.onBeforeEnvironmentCompile(({ environment }) => {
25+
const envName = environment.name;
26+
const config = api.getNormalizedConfig({
27+
environment: envName,
28+
});
29+
30+
const swc = config.tools.swc;
31+
const tsconfigJsx = compilerOptions?.jsx;
32+
// @ts-ignore
33+
const swcJsx = swc.jsc.transform.react.runtime;
34+
const mapped = mapTsconfigJsxToSwcJsx(tsconfigJsx);
35+
if (swcJsx && mapped !== null) {
36+
const matched = mapped === swcJsx;
37+
if (!matched) {
38+
logger.warn(
39+
`You're using ${color.green(`"${swcJsx}"`)} JSX runtime in SWC, but ${color.green(`"${mapped}"`)} in tsconfig.json. This may cause unexpected behavior, considering aligning them.`,
40+
);
41+
}
42+
}
43+
});
44+
},
45+
});
46+
47+
export const composeCheckConfig = (
48+
compilerOptions: Record<string, any>,
49+
): RsbuildConfig => {
50+
return { plugins: [checkJsx({ compilerOptions })] };
51+
};

packages/core/src/config.ts

+4
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@ import {
1414
} from '@rsbuild/core';
1515
import { glob } from 'tinyglobby';
1616
import { composeAssetConfig } from './asset/assetConfig';
17+
import { composeCheckConfig } from './check';
1718
import {
1819
DEFAULT_CONFIG_EXTENSIONS,
1920
DEFAULT_CONFIG_NAME,
@@ -1327,6 +1328,8 @@ async function composeLibRsbuildConfig(
13271328
rootPath,
13281329
config.source?.tsconfigPath,
13291330
);
1331+
1332+
const checkConfig = composeCheckConfig({ compilerOptions });
13301333
const cssModulesAuto = config.output?.cssModules?.auto ?? true;
13311334

13321335
const {
@@ -1438,6 +1441,7 @@ async function composeLibRsbuildConfig(
14381441
dtsConfig,
14391442
bannerFooterConfig,
14401443
decoratorsConfig,
1444+
checkConfig,
14411445
);
14421446
}
14431447

packages/core/src/utils/helper.ts

+14-12
Original file line numberDiff line numberDiff line change
@@ -167,21 +167,23 @@ export function omit<T extends object, U extends keyof T>(
167167
);
168168
}
169169

170-
export function isPluginIncluded(
170+
function findPlugin(pluginName: string, plugins?: RsbuildPlugins) {
171+
return plugins?.find((plugin) => {
172+
if (Array.isArray(plugin)) {
173+
return isPluginIncluded(pluginName, plugin);
174+
}
175+
if (typeof plugin === 'object' && plugin !== null && 'name' in plugin) {
176+
return plugin.name === pluginName;
177+
}
178+
return false;
179+
});
180+
}
181+
182+
function isPluginIncluded(
171183
pluginName: string,
172184
plugins?: RsbuildPlugins,
173185
): boolean {
174-
return Boolean(
175-
plugins?.some((plugin) => {
176-
if (Array.isArray(plugin)) {
177-
return isPluginIncluded(pluginName, plugin);
178-
}
179-
if (typeof plugin === 'object' && plugin !== null && 'name' in plugin) {
180-
return plugin.name === pluginName;
181-
}
182-
return false;
183-
}),
184-
);
186+
return Boolean(findPlugin(pluginName, plugins));
185187
}
186188

187189
export function checkMFPlugin(

pnpm-lock.yaml

+12
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

tests/integration/check/index.test.ts

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
import { join } from 'node:path';
2+
import stripAnsi from 'strip-ansi';
3+
import { buildAndGetResults, proxyConsole } from 'test-helper';
4+
import { expect, test } from 'vitest';
5+
6+
test('should receive JSX mismatch warning of SWC with tsconfig', async () => {
7+
const { logs, restore } = proxyConsole();
8+
const fixturePath = join(__dirname, 'jsx');
9+
await buildAndGetResults({ fixturePath });
10+
const logStrings = logs
11+
.map((log) => stripAnsi(log))
12+
.filter((log) => log.startsWith('warn'))
13+
.sort()
14+
.join('\n');
15+
16+
expect(logStrings).toMatchInlineSnapshot(`
17+
"warn You're using "automatic" JSX runtime in SWC, but "classic" in tsconfig.json. This may cause unexpected behavior, considering aligning them.
18+
warn You're using "automatic" JSX runtime in SWC, but "classic" in tsconfig.json. This may cause unexpected behavior, considering aligning them."
19+
`);
20+
21+
restore();
22+
});
+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
{
2+
"name": "check-jsx-test",
3+
"version": "1.0.0",
4+
"private": true,
5+
"type": "module",
6+
"devDependencies": {
7+
"@rsbuild/plugin-react": "^1.1.0",
8+
"@types/react": "^19.0.6",
9+
"react": "^19.0.0"
10+
}
11+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
import { pluginReact } from '@rsbuild/plugin-react';
2+
import { defineConfig } from '@rslib/core';
3+
import { generateBundleCjsConfig, generateBundleEsmConfig } from 'test-helper';
4+
5+
export default defineConfig({
6+
lib: [generateBundleEsmConfig(), generateBundleCjsConfig()],
7+
plugins: [pluginReact()],
8+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
import React from 'react';
2+
3+
export const Foo = <div>foo</div>;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
{
2+
"extends": "@rslib/tsconfig/base",
3+
"compilerOptions": {
4+
"baseUrl": "./",
5+
"jsx": "react"
6+
},
7+
"include": ["src"]
8+
}

0 commit comments

Comments
 (0)