diff --git a/.eslintrc.js b/.eslintrc.js
index bf4642bb..5edc3d8f 100644
--- a/.eslintrc.js
+++ b/.eslintrc.js
@@ -51,6 +51,17 @@ module.exports = {
'no-console': 'off',
},
},
+ {
+ files: ['packages/hmr/lib/runtime/*.ts'],
+ rules: {
+ 'no-console': 'off',
+ '@typescript-eslint/no-explicit-any': 'off',
+ '@typescript-eslint/no-unsafe-call': 'off',
+ '@typescript-eslint/no-unsafe-member-access': 'off',
+ '@typescript-eslint/no-unsafe-argument': 'off',
+ '@typescript-eslint/no-unsafe-assignment': 'off',
+ },
+ },
{
files: ['packages/jest/lib/**/*.ts'],
rules: {
diff --git a/.yarn/cache/@types-react-refresh-npm-0.14.3-d201a15407-ff48c18f92.zip b/.yarn/cache/@types-react-refresh-npm-0.14.3-d201a15407-ff48c18f92.zip
new file mode 100644
index 00000000..6bf799ac
Binary files /dev/null and b/.yarn/cache/@types-react-refresh-npm-0.14.3-d201a15407-ff48c18f92.zip differ
diff --git a/.yarn/cache/swc-plugin-global-module-npm-0.1.0-alpha.5-cd339b699a-10523c5464.zip b/.yarn/cache/swc-plugin-global-module-npm-0.1.0-alpha.5-cd339b699a-10523c5464.zip
new file mode 100644
index 00000000..49c0da61
Binary files /dev/null and b/.yarn/cache/swc-plugin-global-module-npm-0.1.0-alpha.5-cd339b699a-10523c5464.zip differ
diff --git a/README.md b/README.md
index 0f40b9ff..4168443b 100644
--- a/README.md
+++ b/README.md
@@ -28,7 +28,7 @@
- 💾 In-memory & Local File System Caching
- 🎨 Flexible & Extensible
- 🔥 Supports JSC & Hermes Runtime
-- 🔄 Supports Live Reload
+- 🔄 Supports HMR & Live Reload
- 🐛 Supports Debugging(Flipper, Chrome Debugger)
- 🌍 Supports All Platforms(Android, iOS, Web)
- ✨ New Architecture Ready
diff --git a/docs/pages/configuration/basic-configuration.mdx b/docs/pages/configuration/basic-configuration.mdx
index 8c5d10b8..c135d96a 100644
--- a/docs/pages/configuration/basic-configuration.mdx
+++ b/docs/pages/configuration/basic-configuration.mdx
@@ -16,6 +16,9 @@ exports.default = {};
By default, follow the configuration below.
```js
+/**
+ * @type {import('@react-native-esbuild/core').Config}
+ */
exports.default = {
cache: true,
logger: {
@@ -28,13 +31,6 @@ exports.default = {
assetExtensions: [/* internal/lib/defaults.ts */],
},
transformer: {
- jsc: {
- transform: {
- react: {
- runtime: 'automatic',
- },
- },
- },
stripFlowPackageNames: ['react-native'],
},
web: {
@@ -68,7 +64,6 @@ Resolver configurations.
Transformer configurations.
-- `transformer.jsc`: [jsc](https://swc.rs/docs/configuration/compilation) config in swc.
- `transformer.stripFlowPackageNames`: Package names to strip flow syntax from (Defaults to `['react-native']`)
- `transformer.fullyTransformPackageNames`: Package names to fully transform with [metro-react-native-babel-preset](https://github.com/facebook/react-native/tree/main/packages/react-native-babel-preset) from
- `transformer.additionalTransformRules`: Additional transform rules. This rules will be applied before phase of transform to es5
@@ -90,6 +85,15 @@ Additional Esbuild plugins.
For more details, go to [Custom Plugins](/configuration/custom-plugins)
+### experimental
+
+
+ Experimental configurations.
+
+
+- `experimental.hmr`: Enable HMR(Hot Module Replacement) on development mode. (Defaults to `false`)
+ - For more details and limitations, go to [Hot Module Replacement](/limitations/hot-module-replacement).
+
## Types
@@ -209,6 +213,17 @@ interface Config {
* Additional Esbuild plugins.
*/
plugins?: EsbuildPlugin[];
+ /**
+ * Experimental configurations
+ */
+ experimental?: {
+ /**
+ * Enable HMR(Hot Module Replacement) on development mode.
+ *
+ * Defaults to `false`.
+ */
+ hmr?: boolean;
+ };
/**
* Client event receiver
*/
diff --git a/docs/pages/getting-started/installation.md b/docs/pages/getting-started/installation.md
index 530dc5de..78e783a7 100644
--- a/docs/pages/getting-started/installation.md
+++ b/docs/pages/getting-started/installation.md
@@ -31,7 +31,7 @@ And create `react-native-esbuild.js` to project root.
exports.default = {};
```
-for more details, go to [Configuration](/configuration/basic).
+for more details, go to [Basic Configuration](/configuration/basic-configuration).
## Native Setup
diff --git a/docs/pages/limitations/hot-module-replacement.md b/docs/pages/limitations/hot-module-replacement.md
deleted file mode 100644
index 9cdb6133..00000000
--- a/docs/pages/limitations/hot-module-replacement.md
+++ /dev/null
@@ -1,5 +0,0 @@
-# Hot Module Replacement
-
-Esbuild doesn't currently support Hot Module Replacement(HMR).
-
-Metro is implementing HMR capabilities based on [react-refresh](https://www.npmjs.com/package/react-refresh). I'll be looking at working with this, but as it is one of the complex implementations, unfortunately not sure when it will be available.
diff --git a/docs/pages/limitations/hot-module-replacement.mdx b/docs/pages/limitations/hot-module-replacement.mdx
new file mode 100644
index 00000000..a494cd39
--- /dev/null
+++ b/docs/pages/limitations/hot-module-replacement.mdx
@@ -0,0 +1,30 @@
+import { Callout } from 'nextra/components'
+
+# Hot Module Replacement
+
+
+ HMR(Hot Module Replacement) is experimental.
+
+
+esbuild doesn't currently support Hot Module Replacement(HMR).
+
+So, I working hard for implement custom HMR and it's partially available as an experimental feature.
+
+You can enable HMR by `experimental.hmr` set to `true` in your configuration file.
+
+```js
+/**
+ * @type {import('@react-native-esbuild/core').Config}
+ */
+exports.default = {
+ // ...
+ experimental: {
+ hmr: true,
+ },
+};
+```
+
+and here are some limitations.
+
+- Detects changes in the `/*` only.
+- Changes detected in `/node_modules/*` will be ignored and fully refreshed after rebuild.
diff --git a/example/.gitignore b/example/.gitignore
index f4c3ff65..89fdaefa 100644
--- a/example/.gitignore
+++ b/example/.gitignore
@@ -73,6 +73,9 @@ yarn-error.log
!.yarn/sdks
!.yarn/versions
+# @swc
+.swc
+
# @react-native-esbuild
.rne
.swc
diff --git a/example/.swc/plugins/v7_macos_aarch64_0.104.4/90b69c1e7164e0dc62ebf083a345c7d5b7b0450c10b6dada84547644da60c783 b/example/.swc/plugins/v7_macos_aarch64_0.104.4/90b69c1e7164e0dc62ebf083a345c7d5b7b0450c10b6dada84547644da60c783
deleted file mode 100644
index d5283fd2..00000000
Binary files a/example/.swc/plugins/v7_macos_aarch64_0.104.4/90b69c1e7164e0dc62ebf083a345c7d5b7b0450c10b6dada84547644da60c783 and /dev/null differ
diff --git a/example/react-native-esbuild.config.js b/example/react-native-esbuild.config.js
index d5655063..9547c723 100644
--- a/example/react-native-esbuild.config.js
+++ b/example/react-native-esbuild.config.js
@@ -29,4 +29,7 @@ exports.default = {
],
},
},
+ experimental: {
+ hmr: true,
+ },
};
diff --git a/packages/core/lib/bundler/bundler.ts b/packages/core/lib/bundler/bundler.ts
index 5146daa3..7dc610f6 100644
--- a/packages/core/lib/bundler/bundler.ts
+++ b/packages/core/lib/bundler/bundler.ts
@@ -1,4 +1,5 @@
import path from 'node:path';
+import type { Stats } from 'node:fs';
import esbuild, {
type BuildOptions,
type BuildResult,
@@ -7,6 +8,7 @@ import esbuild, {
import invariant from 'invariant';
import ora from 'ora';
import { getGlobalVariables } from '@react-native-esbuild/internal';
+import { HmrTransformer } from '@react-native-esbuild/hmr';
import {
setEnvironment,
combineWithDefaultBundleOptions,
@@ -35,23 +37,28 @@ import { createBuildStatusPlugin, createMetafilePlugin } from './plugins';
import { BundlerEventEmitter } from './events';
import {
loadConfig,
- getConfigFromGlobal,
createPromiseHandler,
+ getConfigFromGlobal,
getTransformedPreludeScript,
getResolveExtensionsOption,
getLoaderOption,
getEsbuildWebConfig,
+ getExternalFromPackageJson,
+ getExternalModulePattern,
} from './helpers';
import { printLogo, printVersion } from './logo';
export class ReactNativeEsbuildBundler extends BundlerEventEmitter {
- public static caches = new CacheStorage();
- public static shared = new SharedStorage();
+ public static caches = CacheStorage.getInstance();
+ public static shared = SharedStorage.getInstance();
+ private static hmr = new Map();
private appLogger = new Logger('app', LogLevel.Trace);
private buildTasks = new Map();
private plugins: ReactNativeEsbuildPluginCreator[] = [];
- private initialized = false;
private config: Config;
+ private external: string[];
+ private externalPattern: string;
+ private initialized = false;
/**
* Must be bootstrapped first at the entry point
@@ -98,6 +105,11 @@ export class ReactNativeEsbuildBundler extends BundlerEventEmitter {
constructor(private root: string = process.cwd()) {
super();
this.config = getConfigFromGlobal();
+ this.external = getExternalFromPackageJson(root);
+ this.externalPattern = getExternalModulePattern(
+ this.external,
+ this.config.resolver?.assetExtensions ?? [],
+ );
this.on('report', (event) => {
this.broadcastToReporter(event);
});
@@ -131,19 +143,7 @@ export class ReactNativeEsbuildBundler extends BundlerEventEmitter {
private startWatcher(): Promise {
return FileSystemWatcher.getInstance()
- .setHandler((event, changedFile, stats) => {
- const hasTask = this.buildTasks.size > 0;
- ReactNativeEsbuildBundler.shared.setValue({
- watcher: {
- changed: hasTask && event === 'change' ? changedFile : null,
- stats,
- },
- });
-
- for (const { context, handler } of this.buildTasks.values()) {
- context.rebuild().catch((error) => handler?.rejecter?.(error));
- }
- })
+ .setHandler(this.handleFileChanged.bind(this))
.watch(this.root);
}
@@ -156,11 +156,13 @@ export class ReactNativeEsbuildBundler extends BundlerEventEmitter {
invariant(config.resolver, 'invalid resolver configuration');
invariant(config.resolver.mainFields, 'invalid mainFields');
invariant(config.transformer, 'invalid transformer configuration');
- invariant(config.resolver.assetExtensions, 'invalid assetExtension');
+ invariant(config.resolver.assetExtensions, 'invalid assetExtensions');
invariant(config.resolver.sourceExtensions, 'invalid sourceExtensions');
-
setEnvironment(bundleOptions.dev);
+ const enableHmr = Boolean(
+ mode === 'watch' && bundleOptions.dev && config.experimental?.hmr,
+ );
const webSpecifiedOptions =
bundleOptions.platform === 'web'
? getEsbuildWebConfig(mode, this.root, bundleOptions)
@@ -176,6 +178,8 @@ export class ReactNativeEsbuildBundler extends BundlerEventEmitter {
id: this.identifyTaskByBundleOptions(bundleOptions),
root: this.root,
config: this.config,
+ externalPattern: this.externalPattern,
+ enableHmr,
mode,
additionalData,
};
@@ -183,7 +187,7 @@ export class ReactNativeEsbuildBundler extends BundlerEventEmitter {
return {
entryPoints: [bundleOptions.entry],
outfile: bundleOptions.outfile,
- sourceRoot: path.dirname(bundleOptions.entry),
+ sourceRoot: this.root,
mainFields: config.resolver.mainFields,
resolveExtensions: getResolveExtensionsOption(
bundleOptions,
@@ -193,7 +197,13 @@ export class ReactNativeEsbuildBundler extends BundlerEventEmitter {
loader: getLoaderOption(config.resolver.assetExtensions),
define: getGlobalVariables(bundleOptions),
banner: {
- js: await getTransformedPreludeScript(bundleOptions, this.root),
+ js: await getTransformedPreludeScript(
+ bundleOptions,
+ this.root,
+ [enableHmr ? 'swc-plugin-global-module/runtime' : undefined].filter(
+ Boolean,
+ ) as string[],
+ ),
},
plugins: [
createBuildStatusPlugin(context, {
@@ -207,7 +217,7 @@ export class ReactNativeEsbuildBundler extends BundlerEventEmitter {
// Additional plugins in configuration.
...(config.plugins ?? []),
],
- legalComments: bundleOptions.dev ? 'inline' : 'none',
+ legalComments: 'none',
target: 'es6',
format: 'esm',
supported: {
@@ -225,8 +235,8 @@ export class ReactNativeEsbuildBundler extends BundlerEventEmitter {
logLevel: 'silent',
bundle: true,
sourcemap: true,
+ metafile: true,
minify: bundleOptions.minify,
- metafile: bundleOptions.metafile,
write: mode === 'bundle',
...webSpecifiedOptions,
};
@@ -257,6 +267,8 @@ export class ReactNativeEsbuildBundler extends BundlerEventEmitter {
data: { result: BuildResult; success: boolean },
context: PluginContext,
): void {
+ invariant(data.result.metafile, 'invalid metafile');
+
/**
* Exit at the end of a build in bundle mode.
*
@@ -267,8 +279,19 @@ export class ReactNativeEsbuildBundler extends BundlerEventEmitter {
process.exit(1);
}
+ const sharedStorage = ReactNativeEsbuildBundler.shared.get(context.id);
+ const hmrController = ReactNativeEsbuildBundler.hmr.get(context.id);
const currentTask = this.buildTasks.get(context.id);
+ invariant(sharedStorage, 'invalid shared storage');
invariant(currentTask, 'no task');
+
+ if (context.enableHmr) {
+ invariant(hmrController, 'no hmr controller');
+ ReactNativeEsbuildBundler.shared.setValue({
+ bundleMeta: HmrTransformer.createBundleMeta(data.result.metafile),
+ });
+ }
+
const bundleEndedAt = new Date();
const bundleFilename = context.outfile;
const bundleSourcemapFilename = `${bundleFilename}.map`;
@@ -310,16 +333,53 @@ export class ReactNativeEsbuildBundler extends BundlerEventEmitter {
revisionId,
id: context.id,
additionalData: context.additionalData,
+ update: null,
+ });
+ }
+ }
+
+ private handleFileChanged(
+ event: string,
+ changedFile: string,
+ _stats?: Stats,
+ ): void {
+ const hasTask = this.buildTasks.size > 0;
+ const isChanged = event === 'change';
+ if (!(hasTask && isChanged)) return;
+
+ if (
+ this.config.experimental?.hmr &&
+ HmrTransformer.isBoundary(changedFile)
+ ) {
+ for (const [
+ id,
+ hmrController,
+ ] of ReactNativeEsbuildBundler.hmr.entries()) {
+ const { bundleMeta } = ReactNativeEsbuildBundler.shared.get(id);
+ Promise.resolve(
+ bundleMeta ? hmrController.getDelta(changedFile, bundleMeta) : null,
+ ).then((update) => {
+ this.emit('build-end', {
+ id,
+ update,
+ revisionId: new Date().getTime().toString(),
+ });
+ });
+ }
+ } else {
+ this.buildTasks.forEach(({ context }) => {
+ context.rebuild();
});
}
}
- private async getOrCreateBundleTask(
+ private async getOrSetupTask(
bundleOptions: BundleOptions,
additionalData?: BundlerAdditionalData,
): Promise {
const targetTaskId = this.identifyTaskByBundleOptions(bundleOptions);
+ // Build Task
if (!this.buildTasks.has(targetTaskId)) {
logger.debug(`bundle task not registered (id: ${targetTaskId})`);
const buildOptions = await this.getBuildOptionsForBundler(
@@ -340,6 +400,35 @@ export class ReactNativeEsbuildBundler extends BundlerEventEmitter {
logger.debug(`bundle task is now watching (id: ${targetTaskId})`);
}
+ // HMR Transformer
+ if (
+ this.config.experimental?.hmr &&
+ !ReactNativeEsbuildBundler.hmr.has(targetTaskId)
+ ) {
+ const {
+ stripFlowPackageNames,
+ fullyTransformPackageNames,
+ additionalTransformRules,
+ } = this.config.transformer ?? {};
+ ReactNativeEsbuildBundler.hmr.set(
+ targetTaskId,
+ new HmrTransformer(
+ {
+ ...bundleOptions,
+ id: targetTaskId,
+ root: this.root,
+ externalPattern: this.externalPattern,
+ },
+ {
+ additionalBabelRules: additionalTransformRules?.babel,
+ additionalSwcRules: additionalTransformRules?.swc,
+ fullyTransformPackageNames,
+ stripFlowPackageNames,
+ },
+ ),
+ );
+ }
+
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion -- Already `set()` if not exist.
return this.buildTasks.get(targetTaskId)!;
}
@@ -426,7 +515,7 @@ export class ReactNativeEsbuildBundler extends BundlerEventEmitter {
throw new Error('serve mode is only available on web platform');
}
- const buildTask = await this.getOrCreateBundleTask(
+ const buildTask = await this.getOrSetupTask(
combineWithDefaultBundleOptions(bundleOptions),
additionalData,
);
@@ -442,7 +531,7 @@ export class ReactNativeEsbuildBundler extends BundlerEventEmitter {
additionalData?: BundlerAdditionalData,
): Promise {
this.throwIfNotInitialized();
- const buildTask = await this.getOrCreateBundleTask(
+ const buildTask = await this.getOrSetupTask(
combineWithDefaultBundleOptions(bundleOptions),
additionalData,
);
diff --git a/packages/core/lib/bundler/events/index.ts b/packages/core/lib/bundler/events/index.ts
index cd3718b7..27473fe6 100644
--- a/packages/core/lib/bundler/events/index.ts
+++ b/packages/core/lib/bundler/events/index.ts
@@ -1,4 +1,5 @@
import EventEmitter from 'node:events';
+import type { BundleUpdate } from '@react-native-esbuild/hmr';
import type {
BundlerAdditionalData,
BuildStatus,
@@ -46,6 +47,7 @@ export interface BundlerEventPayload {
'build-end': {
id: number;
revisionId: string;
+ update: BundleUpdate | null;
additionalData?: BundlerAdditionalData;
};
'build-status-change': BuildStatus & {
diff --git a/packages/core/lib/bundler/helpers/config.ts b/packages/core/lib/bundler/helpers/config.ts
index 8c83fd43..c04ce8e3 100644
--- a/packages/core/lib/bundler/helpers/config.ts
+++ b/packages/core/lib/bundler/helpers/config.ts
@@ -29,13 +29,6 @@ export const loadConfig = (configFilePath?: string): Config => {
assetExtensions: ASSET_EXTENSIONS,
},
transformer: {
- jsc: {
- transform: {
- react: {
- runtime: 'automatic',
- },
- },
- },
stripFlowPackageNames: ['react-native'],
},
web: {
diff --git a/packages/core/lib/bundler/helpers/fs.ts b/packages/core/lib/bundler/helpers/fs.ts
new file mode 100644
index 00000000..2c774537
--- /dev/null
+++ b/packages/core/lib/bundler/helpers/fs.ts
@@ -0,0 +1,14 @@
+import fs from 'node:fs';
+import path from 'node:path';
+
+export const getExternalFromPackageJson = (root: string): string[] => {
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment -- allow any.
+ const { dependencies = {} } = JSON.parse(
+ fs.readFileSync(path.join(root, 'package.json'), 'utf-8'),
+ );
+ return [
+ 'react/jsx-runtime',
+ '@react-navigation/devtools',
+ ...Object.keys(dependencies as Record),
+ ];
+};
diff --git a/packages/core/lib/bundler/helpers/index.ts b/packages/core/lib/bundler/helpers/index.ts
index dd49165c..ad840df1 100644
--- a/packages/core/lib/bundler/helpers/index.ts
+++ b/packages/core/lib/bundler/helpers/index.ts
@@ -1,3 +1,4 @@
export * from './async';
export * from './config';
+export * from './fs';
export * from './internal';
diff --git a/packages/core/lib/bundler/helpers/internal.ts b/packages/core/lib/bundler/helpers/internal.ts
index 5955e7f3..eccde4d7 100644
--- a/packages/core/lib/bundler/helpers/internal.ts
+++ b/packages/core/lib/bundler/helpers/internal.ts
@@ -1,16 +1,21 @@
+import fs from 'node:fs/promises';
import type { BuildOptions } from 'esbuild';
import { getPreludeScript } from '@react-native-esbuild/internal';
import type { TransformerContext } from '@react-native-esbuild/transformer';
import {
stripFlowWithSucrase,
- minifyWithSwc,
+ transformWithSwc,
swcPresets,
} from '@react-native-esbuild/transformer';
import type { BundleOptions } from '@react-native-esbuild/config';
+const loadScript = (path: string): Promise =>
+ fs.readFile(require.resolve(path), 'utf-8');
+
export const getTransformedPreludeScript = async (
bundleOptions: BundleOptions,
root: string,
+ additionalScriptPaths?: string[],
): Promise => {
// Dummy context
const context: TransformerContext = {
@@ -20,19 +25,28 @@ export const getTransformedPreludeScript = async (
dev: bundleOptions.dev,
entry: bundleOptions.entry,
};
- const preludeScript = await getPreludeScript(bundleOptions, root);
+
+ const additionalPreludeScripts = await Promise.all(
+ (additionalScriptPaths ?? []).map(loadScript),
+ );
+
+ const preludeScript = [
+ await getPreludeScript(bundleOptions, root),
+ ...additionalPreludeScripts,
+ ].join('\n');
/**
* Remove `"use strict";` added by sucrase.
* @see {@link https://github.com/alangpierce/sucrase/issues/787#issuecomment-1483934492}
*/
- const strippedScript = stripFlowWithSucrase(preludeScript, context)
+ const strippedScript = stripFlowWithSucrase(preludeScript, { context })
.replace(/"use strict";/, '')
.trim();
- return bundleOptions.minify
- ? minifyWithSwc(strippedScript, context, swcPresets.getMinifyPreset())
- : strippedScript;
+ return transformWithSwc(strippedScript, {
+ context,
+ preset: swcPresets.getMinifyPreset({ minify: bundleOptions.minify }),
+ });
};
export const getResolveExtensionsOption = (
@@ -70,3 +84,24 @@ export const getLoaderOption = (
assetExtensions.map((ext) => [ext, 'file'] as const),
);
};
+
+export const getExternalModulePattern = (
+ externalPackages: string[],
+ assetExtensions: string[],
+): string => {
+ const externalPackagePatterns = externalPackages
+ .map((packageName) => `^${packageName}/?$`)
+ .join('|');
+
+ const assetPatterns = [
+ ...assetExtensions,
+ // `.svg` assets will be handled by `svg-transform-plugin`.
+ '.svg',
+ // `.json` contents will be handled by `react-native-runtime-transform-plugin`.
+ '.json',
+ ]
+ .map((extension) => `${extension}$`)
+ .join('|');
+
+ return `(${externalPackagePatterns}|${assetPatterns})`;
+};
diff --git a/packages/core/lib/bundler/plugins/metafilePlugin/metafilePlugin.ts b/packages/core/lib/bundler/plugins/metafilePlugin/metafilePlugin.ts
index 70bb8c1b..96e17ab1 100644
--- a/packages/core/lib/bundler/plugins/metafilePlugin/metafilePlugin.ts
+++ b/packages/core/lib/bundler/plugins/metafilePlugin/metafilePlugin.ts
@@ -12,18 +12,16 @@ export const createMetafilePlugin: ReactNativeEsbuildPluginCreator = (
name: NAME,
setup: (build): void => {
build.onEnd(async (result: BuildResult) => {
- const { metafile } = result;
+ if (!(context.metafile && result.metafile)) return;
+
const filename = path.join(
context.root,
`metafile-${context.platform}-${new Date().getTime().toString()}.json`,
);
-
- if (metafile) {
- logger.debug('writing esbuild metafile', { destination: filename });
- await fs.writeFile(filename, JSON.stringify(metafile), {
- encoding: 'utf-8',
- });
- }
+ logger.debug('writing esbuild metafile', { destination: filename });
+ await fs.writeFile(filename, JSON.stringify(result.metafile), {
+ encoding: 'utf-8',
+ });
});
},
});
diff --git a/packages/core/lib/bundler/plugins/statusPlugin/StatusLogger.ts b/packages/core/lib/bundler/plugins/statusPlugin/StatusLogger.ts
index 0a8c4384..15d5cd02 100644
--- a/packages/core/lib/bundler/plugins/statusPlugin/StatusLogger.ts
+++ b/packages/core/lib/bundler/plugins/statusPlugin/StatusLogger.ts
@@ -7,10 +7,16 @@ import { getBuildStatusCachePath } from '@react-native-esbuild/config';
import { colors, isTTY } from '@react-native-esbuild/utils';
import { logger } from '../../../shared';
import { ESBUILD_LABEL } from '../../logo';
-import type { BuildStatus, PluginContext } from '../../../types';
+import type {
+ BuildStatus,
+ BundlerSharedData,
+ PluginContext,
+} from '../../../types';
+import { SharedStorage } from '../../storages';
import { fromTemplate, getSummaryTemplate } from './templates';
export class StatusLogger {
+ private bundlerSharedData: BundlerSharedData;
private platformText: string;
private spinner: Ora;
private totalModuleCount = 0;
@@ -19,6 +25,7 @@ export class StatusLogger {
private previousPercent = 0;
constructor(private context: PluginContext) {
+ this.bundlerSharedData = SharedStorage.getInstance().get(context.id);
this.platformText = colors.gray(
`[${[context.platform, context.dev ? 'dev' : null]
.filter(Boolean)
@@ -103,6 +110,7 @@ export class StatusLogger {
this.previousPercent = 0;
this.statusUpdate();
+ process.stdout.write('\n');
isTTY()
? this.spinner.start()
: this.print(`${this.platformText} build in progress...`);
diff --git a/packages/core/lib/bundler/storages/CacheStorage.ts b/packages/core/lib/bundler/storages/CacheStorage.ts
index c62a94c4..b0811218 100644
--- a/packages/core/lib/bundler/storages/CacheStorage.ts
+++ b/packages/core/lib/bundler/storages/CacheStorage.ts
@@ -9,7 +9,16 @@ import { Storage } from './Storage';
const CACHE_DIRECTORY = path.join(os.tmpdir(), GLOBAL_CACHE_DIR);
export class CacheStorage extends Storage {
- constructor() {
+ private static instance: CacheStorage | null = null;
+
+ public static getInstance(): CacheStorage {
+ if (CacheStorage.instance === null) {
+ CacheStorage.instance = new CacheStorage();
+ }
+ return CacheStorage.instance;
+ }
+
+ private constructor() {
super();
try {
fs.accessSync(CACHE_DIRECTORY, fs.constants.R_OK | fs.constants.W_OK);
diff --git a/packages/core/lib/bundler/storages/SharedStorage.ts b/packages/core/lib/bundler/storages/SharedStorage.ts
index 05c2389e..d2f77a40 100644
--- a/packages/core/lib/bundler/storages/SharedStorage.ts
+++ b/packages/core/lib/bundler/storages/SharedStorage.ts
@@ -2,13 +2,21 @@ import type { BundlerSharedData } from '../../types';
import { Storage } from './Storage';
export class SharedStorage extends Storage {
+ private static instance: SharedStorage | null = null;
+
+ public static getInstance(): SharedStorage {
+ if (SharedStorage.instance === null) {
+ SharedStorage.instance = new SharedStorage();
+ }
+ return SharedStorage.instance;
+ }
+
+ private constructor() {
+ super();
+ }
+
private getDefaultSharedData(): BundlerSharedData {
- return {
- watcher: {
- changed: null,
- stats: undefined,
- },
- };
+ return { bundleMeta: undefined };
}
public get(key: number): BundlerSharedData {
@@ -25,17 +33,13 @@ export class SharedStorage extends Storage {
public setValue(value: Partial): void {
for (const sharedData of this.data.values()) {
- sharedData.watcher.changed =
- value.watcher?.changed ?? sharedData.watcher.changed;
- sharedData.watcher.stats =
- value.watcher?.stats ?? sharedData.watcher.stats;
+ sharedData.bundleMeta = value.bundleMeta;
}
}
public clearAll(): Promise {
for (const sharedData of this.data.values()) {
- sharedData.watcher.changed = null;
- sharedData.watcher.stats = undefined;
+ sharedData.bundleMeta = undefined;
}
return Promise.resolve();
}
diff --git a/packages/core/lib/types.ts b/packages/core/lib/types.ts
index 139eb200..21c5e762 100644
--- a/packages/core/lib/types.ts
+++ b/packages/core/lib/types.ts
@@ -1,11 +1,10 @@
-import type { Stats } from 'node:fs';
import type { BuildContext, Plugin } from 'esbuild';
import type {
BabelTransformRule,
SwcTransformRule,
} from '@react-native-esbuild/transformer';
+import type { BundleMeta } from '@react-native-esbuild/hmr';
import type { BundleOptions } from '@react-native-esbuild/config';
-import type { JscConfig } from '@swc/core';
export interface Config {
/**
@@ -58,10 +57,6 @@ export interface Config {
* Transformer configurations
*/
transformer?: {
- /**
- * Swc's `jsc` config.
- */
- jsc: Pick;
/**
* Strip flow syntax.
*
@@ -125,6 +120,17 @@ export interface Config {
* Additional Esbuild plugins.
*/
plugins?: Plugin[];
+ /**
+ * Experimental configurations
+ */
+ experimental?: {
+ /**
+ * Enable HMR(Hot Module Replacement) on development mode.
+ *
+ * Defaults to `false`.
+ */
+ hmr?: boolean;
+ };
/**
* Client event receiver (only work on native)
*/
@@ -167,10 +173,7 @@ export type ReactNativeEsbuildPluginCreator = (
) => Plugin;
export interface BundlerSharedData {
- watcher: {
- changed: string | null;
- stats?: Stats;
- };
+ bundleMeta?: BundleMeta;
}
export type BundlerAdditionalData = Record;
@@ -180,6 +183,8 @@ export interface PluginContext extends BundleOptions {
root: string;
config: Config;
mode: BundleMode;
+ enableHmr: boolean;
+ externalPattern: string;
additionalData?: BundlerAdditionalData;
}
diff --git a/packages/core/package.json b/packages/core/package.json
index 7005cc38..93bc38f3 100644
--- a/packages/core/package.json
+++ b/packages/core/package.json
@@ -41,6 +41,7 @@
},
"dependencies": {
"@react-native-esbuild/config": "workspace:*",
+ "@react-native-esbuild/hmr": "workspace:*",
"@react-native-esbuild/internal": "workspace:*",
"@react-native-esbuild/transformer": "workspace:*",
"@react-native-esbuild/utils": "workspace:*",
@@ -51,7 +52,8 @@
"ora": "^5.4.1"
},
"peerDependencies": {
- "react-native": "*"
+ "react-native": "*",
+ "swc-plugin-global-module": "*"
},
"devDependencies": {
"@babel/core": "^7.23.2",
diff --git a/packages/dev-server/lib/middlewares/hmr.ts b/packages/dev-server/lib/middlewares/hmr.ts
new file mode 100644
index 00000000..42c26f32
--- /dev/null
+++ b/packages/dev-server/lib/middlewares/hmr.ts
@@ -0,0 +1,111 @@
+import {
+ HmrAppServer,
+ HmrWebServer,
+ type HmrClientMessage,
+} from '@react-native-esbuild/hmr';
+import { getReloadByDevSettingsProxy } from '@react-native-esbuild/internal';
+import type { HmrMiddleware } from '../types';
+import { logger } from '../shared';
+
+export const createHmrMiddlewareForApp = ({
+ onMessage,
+}: {
+ onMessage?: (event: HmrClientMessage) => void;
+}): HmrMiddleware => {
+ const server = new HmrAppServer();
+
+ server.setMessageHandler((event) => onMessage?.(event));
+
+ const updateStart = (): void => {
+ logger.debug('send update-start message');
+ server.send('update-start', { isInitialUpdate: false });
+ };
+
+ const updateDone = (): void => {
+ logger.debug('send update-done message');
+ server.send('update-done', undefined);
+ };
+
+ const hotReload = (revisionId: string, code: string): void => {
+ logger.debug('send update message (hmr)');
+ server.send('update', {
+ added: [
+ {
+ module: [-1, code],
+ sourceMappingURL: null,
+ sourceURL: null,
+ },
+ ],
+ deleted: [],
+ modified: [],
+ isInitialUpdate: false,
+ revisionId,
+ });
+ };
+
+ const liveReload = (revisionId: string): void => {
+ logger.debug('send update message (live reload)');
+ server.send('update', {
+ added: [
+ {
+ module: [-1, getReloadByDevSettingsProxy()],
+ sourceMappingURL: null,
+ sourceURL: null,
+ },
+ ],
+ deleted: [],
+ modified: [],
+ isInitialUpdate: false,
+ revisionId,
+ });
+ };
+
+ return { server, updateStart, updateDone, hotReload, liveReload };
+};
+
+export const createHmrMiddlewareForWeb = (): HmrMiddleware => {
+ const server = new HmrWebServer();
+
+ // eslint-disable-next-line @typescript-eslint/no-empty-function -- noop
+ const noop = (): void => {};
+
+ return {
+ server,
+ updateStart: noop,
+ updateDone: noop,
+ hotReload: (_revisionId: string, _code: string): void => {
+ logger.debug('send update message (hmr)');
+ // TODO
+ // server.send('update', {
+ // added: [
+ // {
+ // module: [-1, code],
+ // sourceMappingURL: null,
+ // sourceURL: null,
+ // },
+ // ],
+ // deleted: [],
+ // modified: [],
+ // isInitialUpdate: false,
+ // revisionId,
+ // });
+ },
+ liveReload: (_revisionId: string): void => {
+ logger.debug('send update message (live reload)');
+ // TODO
+ // server.send('update', {
+ // added: [
+ // {
+ // module: [-1, 'window.location.reload();'],
+ // sourceMappingURL: null,
+ // sourceURL: null,
+ // },
+ // ],
+ // deleted: [],
+ // modified: [],
+ // isInitialUpdate: false,
+ // revisionId,
+ // });
+ },
+ };
+};
diff --git a/packages/dev-server/lib/middlewares/hotReload.ts b/packages/dev-server/lib/middlewares/hotReload.ts
deleted file mode 100644
index 3ce6a213..00000000
--- a/packages/dev-server/lib/middlewares/hotReload.ts
+++ /dev/null
@@ -1,124 +0,0 @@
-import { Server, type WebSocket, type MessageEvent, type Data } from 'ws';
-import type { ClientLogEvent } from '@react-native-esbuild/core';
-import { getReloadByDevSettingsProxy } from '@react-native-esbuild/internal';
-import { logger } from '../shared';
-import type {
- HotReloadMiddleware,
- HmrClientMessage,
- HmrUpdateDoneMessage,
- HmrUpdateMessage,
- HmrUpdateStartMessage,
-} from '../types';
-
-const getMessage = (data: Data): HmrClientMessage | null => {
- try {
- // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment -- Web socket data.
- const parsedData = JSON.parse(String(data));
- return 'type' in parsedData ? (parsedData as HmrClientMessage) : null;
- } catch (error) {
- return null;
- }
-};
-
-export const createHotReloadMiddleware = ({
- onLog,
-}: {
- onLog?: (event: ClientLogEvent) => void;
-}): HotReloadMiddleware => {
- const server = new Server({ noServer: true });
- let connectedSocket: WebSocket | null = null;
-
- const handleClose = (): void => {
- connectedSocket = null;
- logger.debug('HMR web socket was closed');
- };
-
- const handleMessage = (event: MessageEvent): void => {
- const message = getMessage(event.data);
- if (!message) return;
-
- /**
- * @see {@link https://github.com/facebook/metro/blob/v0.77.0/packages/metro/src/HmrServer.js#L200-L239}
- */
- switch (message.type) {
- case 'log': {
- onLog?.({
- type: 'client_log',
- level: message.level,
- data: message.data,
- mode: 'BRIDGE',
- });
- break;
- }
-
- // Not supported
- case 'register-entrypoints':
- case 'log-opt-in':
- break;
- }
- };
-
- const handleError = (error?: Error): void => {
- if (error) {
- logger.error('unable to send HMR update message', error);
- }
- };
-
- /**
- * Send reload code to client.
- *
- * @see {@link https://github.com/facebook/metro/blob/v0.77.0/packages/metro-runtime/src/modules/HMRClient.js#L91-L99}
- */
- const hotReload = (revisionId: string): void => {
- const hmrUpdateMessage: HmrUpdateMessage = {
- type: 'update',
- body: {
- added: [
- {
- module: [-1, getReloadByDevSettingsProxy()],
- sourceMappingURL: null,
- sourceURL: null,
- },
- ],
- deleted: [],
- modified: [],
- isInitialUpdate: false,
- revisionId,
- },
- };
-
- logger.debug('sending update message with reload code');
- connectedSocket?.send(JSON.stringify(hmrUpdateMessage), handleError);
- };
-
- const updateStart = (): void => {
- logger.debug('sending update-start');
- const hmrUpdateStartMessage: HmrUpdateStartMessage = {
- type: 'update-start',
- body: {
- isInitialUpdate: false,
- },
- };
- connectedSocket?.send(JSON.stringify(hmrUpdateStartMessage), handleError);
- };
-
- const updateDone = (): void => {
- logger.debug('sending update-done');
- const hmrUpdateDoneMessage: HmrUpdateDoneMessage = { type: 'update-done' };
- connectedSocket?.send(JSON.stringify(hmrUpdateDoneMessage), handleError);
- };
-
- server.on('connection', (socket) => {
- connectedSocket = socket;
- connectedSocket.onerror = handleClose;
- connectedSocket.onclose = handleClose;
- connectedSocket.onmessage = handleMessage;
- logger.debug('HMR web socket was connected');
- });
-
- server.on('error', (error) => {
- logger.error('HMR web socket server error', error);
- });
-
- return { server, hotReload, updateStart, updateDone };
-};
diff --git a/packages/dev-server/lib/middlewares/index.ts b/packages/dev-server/lib/middlewares/index.ts
index 6e88227f..caf48ffb 100644
--- a/packages/dev-server/lib/middlewares/index.ts
+++ b/packages/dev-server/lib/middlewares/index.ts
@@ -1,4 +1,4 @@
-export * from './hotReload';
+export * from './hmr';
export * from './indexPage';
export * from './serveAsset';
export * from './serveBundle';
diff --git a/packages/dev-server/lib/server/ReactNativeAppServer.ts b/packages/dev-server/lib/server/ReactNativeAppServer.ts
index 1a18b19c..fea00129 100644
--- a/packages/dev-server/lib/server/ReactNativeAppServer.ts
+++ b/packages/dev-server/lib/server/ReactNativeAppServer.ts
@@ -7,7 +7,7 @@ import { InspectorProxy } from 'metro-inspector-proxy';
import { createDevServerMiddleware } from '@react-native-community/cli-server-api';
import { ReactNativeEsbuildBundler } from '@react-native-esbuild/core';
import {
- createHotReloadMiddleware,
+ createHmrMiddlewareForApp,
createServeAssetMiddleware,
createServeBundleMiddleware,
createSymbolicateMiddleware,
@@ -105,10 +105,26 @@ export class ReactNativeAppServer extends DevServer {
throw new Error('server is not initialized');
}
- const { server: hotReloadWss, ...hr } = createHotReloadMiddleware({
- onLog: (event) => {
- this.eventsSocketEndpoint.reportEvent(event);
- this.bundler?.emit('report', event);
+ const { server: hmrServer, ...hmr } = createHmrMiddlewareForApp({
+ onMessage: (message) => {
+ switch (message.type) {
+ case 'log': {
+ const clientLogEvent = {
+ type: 'client_log',
+ level: message.level,
+ data: message.data,
+ mode: 'BRIDGE',
+ } as const;
+ this.bundler?.emit('report', clientLogEvent);
+ this.eventsSocketEndpoint.reportEvent(clientLogEvent);
+ break;
+ }
+
+ // not supported
+ case 'register-entrypoints':
+ case 'log-opt-in':
+ break;
+ }
},
});
@@ -123,20 +139,22 @@ export class ReactNativeAppServer extends DevServer {
);
const webSocketServer: Record = {
- '/hot': hotReloadWss,
+ '/hot': hmrServer.getWebSocketServer(),
'/debugger-proxy': this.debuggerProxyEndpoint.server,
'/message': this.messageSocketEndpoint.server,
'/events': this.eventsSocketEndpoint.server,
...inspectorProxyWss,
};
- this.bundler.on('build-start', hr.updateStart);
- this.bundler.on('build-end', ({ revisionId, additionalData }) => {
+ this.bundler.on('build-start', hmr.updateStart);
+ this.bundler.on('build-end', ({ revisionId, update, additionalData }) => {
// `additionalData` can be `{ disableRefresh: true }` by `serve-asset-middleware`.
if (!additionalData?.disableRefresh) {
- hr.hotReload(revisionId);
+ update === null || update.fullyReload
+ ? hmr.liveReload(revisionId)
+ : hmr.hotReload(revisionId, update.code);
}
- hr.updateDone();
+ hmr.updateDone();
});
this.server.on('upgrade', (request, socket, head) => {
diff --git a/packages/dev-server/lib/server/ReactNativeWebServer.ts b/packages/dev-server/lib/server/ReactNativeWebServer.ts
index c9fd5f1e..de5291ef 100644
--- a/packages/dev-server/lib/server/ReactNativeWebServer.ts
+++ b/packages/dev-server/lib/server/ReactNativeWebServer.ts
@@ -3,6 +3,7 @@ import http, {
type ServerResponse,
type IncomingMessage,
} from 'node:http';
+import { parse } from 'node:url';
import type { ServeResult } from 'esbuild';
import invariant from 'invariant';
import { ReactNativeEsbuildBundler } from '@react-native-esbuild/core';
@@ -10,7 +11,10 @@ import {
combineWithDefaultBundleOptions,
type BundleOptions,
} from '@react-native-esbuild/config';
-import { createSymbolicateMiddleware } from '../middlewares';
+import {
+ createHmrMiddlewareForWeb,
+ createSymbolicateMiddleware,
+} from '../middlewares';
import { logger } from '../shared';
import type { DevServerOptions } from '../types';
import { DevServer } from './DevServer';
@@ -96,6 +100,16 @@ export class ReactNativeWebServer extends DevServer {
this.devServerOptions.root,
).initialize({ watcherEnabled: true }));
+ const { server: hmrServer, ...hmr } = createHmrMiddlewareForWeb();
+
+ this.bundler.on('build-end', ({ revisionId, update, additionalData }) => {
+ if (!additionalData?.disableRefresh) {
+ update?.fullyReload
+ ? hmr.liveReload(revisionId)
+ : hmr.hotReload(revisionId, update?.code ?? '');
+ }
+ });
+
const symbolicateMiddleware = createSymbolicateMiddleware(
{ bundler, devServerOptions: this.devServerOptions },
{ webBundleOptions: this.bundleOptions },
@@ -114,6 +128,20 @@ export class ReactNativeWebServer extends DevServer {
});
});
+ this.server.on('upgrade', (request, socket, head) => {
+ if (!request.url) return;
+ const { pathname } = parse(request.url);
+
+ if (pathname === '/hot') {
+ const wss = hmrServer.getWebSocketServer();
+ wss.handleUpgrade(request, socket, head, (client) => {
+ wss.emit('connection', client, request);
+ });
+ } else {
+ socket.destroy();
+ }
+ });
+
await onPostSetup?.(bundler);
return this;
diff --git a/packages/dev-server/lib/types.ts b/packages/dev-server/lib/types.ts
index 36ac5c11..0c720f9d 100644
--- a/packages/dev-server/lib/types.ts
+++ b/packages/dev-server/lib/types.ts
@@ -1,6 +1,6 @@
import type { IncomingMessage, ServerResponse } from 'node:http';
-import type { Server as WebSocketServer } from 'ws';
import type { ReactNativeEsbuildBundler } from '@react-native-esbuild/core';
+import type { HmrServer } from '@react-native-esbuild/hmr';
export enum BundleRequestType {
Unknown,
@@ -30,11 +30,12 @@ export type DevServerMiddleware = (
next: (error?: unknown) => void,
) => void;
-export interface HotReloadMiddleware {
- server: WebSocketServer;
- hotReload: (revisionId: string) => void;
+export interface HmrMiddleware {
+ server: HmrServer;
updateStart: () => void;
updateDone: () => void;
+ hotReload: (revisionId: string, code: string) => void;
+ liveReload: (revisionId: string) => void;
}
/**
diff --git a/packages/dev-server/package.json b/packages/dev-server/package.json
index ed84c8a9..24955fdf 100644
--- a/packages/dev-server/package.json
+++ b/packages/dev-server/package.json
@@ -49,6 +49,7 @@
"@react-native-community/cli-server-api": "^11.3.6",
"@react-native-esbuild/config": "workspace:*",
"@react-native-esbuild/core": "workspace:*",
+ "@react-native-esbuild/hmr": "workspace:*",
"@react-native-esbuild/internal": "workspace:*",
"@react-native-esbuild/symbolicate": "workspace:*",
"@react-native-esbuild/utils": "workspace:*",
diff --git a/packages/hmr/README.md b/packages/hmr/README.md
new file mode 100644
index 00000000..3c587cde
--- /dev/null
+++ b/packages/hmr/README.md
@@ -0,0 +1,52 @@
+# `@react-native-esbuild/hmr`
+
+> `react-refresh` based HMR implementation for @react-native-esbuild
+
+## Usage
+
+1. Add import statement to top of entry file(`import 'hmr:runtime';`)
+2. Wrap React component module with `wrapWithHmrBoundary`
+
+```js
+import {
+ getHmrRuntimeInitializeScript,
+ wrapWithHmrBoundary,
+ HMR_RUNTIME_IMPORT_NAME
+} from '@react-native-esbuild/hmr';
+
+// In esbuild plugin
+build.onResolve({ filter: new RegExp(HMR_RUNTIME_IMPORT_NAME) }, (args) => {
+ return {
+ path: args.path,
+ namespace: 'hmr-runtime',
+ };
+});
+
+build.onLoad({ filter: /(?:.*)/, namespace: 'hmr-runtime' }, (args) => {
+ return {
+ js: await getHmrRuntimeInitializeScript(),
+ loader: 'js',
+ };
+});
+
+build.onLoad({ filter: /* some filter */ }, (args) => {
+ if (isEntryFile) {
+ code = await fs.readFile(args.path, 'utf-8');
+ code = `import ${HMR_RUNTIME_IMPORT_NAME};\n\n` + code;
+ // ...
+
+ return {
+ contents: code,
+ loader: 'js',
+ };
+ } else {
+ // ...
+
+ return {
+ // code, component name, module id
+ contents: wrapWithHmrBoundary(code, '', args.path),
+ loader: 'js',
+ };
+ }
+});
+```
diff --git a/packages/hmr/build/index.js b/packages/hmr/build/index.js
new file mode 100644
index 00000000..9ea82862
--- /dev/null
+++ b/packages/hmr/build/index.js
@@ -0,0 +1,25 @@
+const path = require('node:path');
+const esbuild = require('esbuild');
+const { getEsbuildBaseOptions, getPackageRoot } = require('../../../shared');
+const { name, version } = require('../package.json');
+
+const buildOptions = getEsbuildBaseOptions(__dirname);
+const root = getPackageRoot(__dirname);
+
+(async () => {
+ // package
+ await esbuild.build(buildOptions);
+
+ // runtime
+ await esbuild.build({
+ entryPoints: [path.join(root, './lib/runtime/setup.ts')],
+ outfile: 'dist/runtime.js',
+ bundle: true,
+ banner: {
+ js: `// ${name}@${version} runtime`,
+ },
+ });
+})().catch((error) => {
+ console.error(error);
+ process.exit(1);
+});
diff --git a/packages/hmr/lib/HmrTransformer.ts b/packages/hmr/lib/HmrTransformer.ts
new file mode 100644
index 00000000..0e92c446
--- /dev/null
+++ b/packages/hmr/lib/HmrTransformer.ts
@@ -0,0 +1,298 @@
+import fs from 'node:fs/promises';
+import path from 'node:path';
+import { performance } from 'node:perf_hooks';
+import type { Metafile } from 'esbuild';
+import { getReloadByDevSettingsProxy } from '@react-native-esbuild/internal';
+import {
+ AsyncTransformPipeline,
+ swcPresets,
+} from '@react-native-esbuild/transformer';
+import type { TransformerContext } from '@react-native-esbuild/transformer';
+import { colors } from '@react-native-esbuild/utils';
+import { logger } from './shared';
+import {
+ HMR_REGISTER_FUNCTION,
+ HMR_UPDATE_FUNCTION,
+ REACT_REFRESH_REGISTER_FUNCTION,
+ REACT_REFRESH_SIGNATURE_FUNCTION,
+} from './constants';
+import type {
+ BundleMeta,
+ BundleUpdate,
+ ModuleInfo,
+ PipelineBuilderOptions,
+} from './types';
+
+const DUMMY_ESBUILD_ARGS = {
+ namespace: '',
+ suffix: '',
+ pluginData: undefined,
+} as const;
+
+export class HmrTransformer {
+ private static boundaryIndex = 0;
+ private stripRootRegex: RegExp;
+ private externalPatternRegex: RegExp;
+ private pipeline: AsyncTransformPipeline;
+
+ public static isBoundary(path: string): boolean {
+ // `runtime.js`: To avoid wrong HMR behavior in monorepo.
+ return !path.includes('/node_modules/') && !path.endsWith('runtime.js');
+ }
+
+ public static asBoundary(id: string, code: string): string {
+ const ident = `__hmr${HmrTransformer.boundaryIndex++}`;
+ return `var ${ident} = ${HMR_REGISTER_FUNCTION}(${JSON.stringify(id)});
+ ${code}
+ ${ident}.dispose(function () { });
+ ${ident}.accept(function (payload) {
+ global.__hmr.reactRefresh.performReactRefresh();
+ });`;
+ }
+
+ public static registerAsExternalModule(
+ id: string,
+ code: string,
+ identifier: string,
+ ): string {
+ return `${code}\nglobal.__modules.external(${JSON.stringify(
+ id,
+ )}, ${identifier});`;
+ }
+
+ constructor(
+ private context: Omit & {
+ externalPattern: string;
+ },
+ builderOptions: PipelineBuilderOptions,
+ ) {
+ const {
+ fullyTransformPackageNames = [],
+ stripFlowPackageNames = [],
+ additionalBabelRules = [],
+ additionalSwcRules = [],
+ } = builderOptions;
+ this.stripRootRegex = new RegExp(`^${context.root}/?`);
+ this.externalPatternRegex = new RegExp(context.externalPattern);
+ this.pipeline = new AsyncTransformPipeline.builder(context)
+ .setSwcPreset(
+ swcPresets.getReactNativeRuntimePreset({
+ experimental: {
+ hmr: {
+ runtime: true,
+ refreshReg: REACT_REFRESH_REGISTER_FUNCTION,
+ refreshSig: REACT_REFRESH_SIGNATURE_FUNCTION,
+ },
+ },
+ }),
+ )
+ .setFullyTransformPackages(fullyTransformPackageNames)
+ .setStripFlowPackages(stripFlowPackageNames)
+ .setAdditionalBabelTransformRules(additionalBabelRules)
+ .setAdditionalSwcTransformRules(additionalSwcRules)
+ .build();
+ }
+
+ /**
+ * Enhance `esbuild.Metafile` to `BundleMeta`.
+ *
+ * Add `parents` property into each module info.
+ * This property will be used as indexed value for explore parent modules.
+ */
+ public static createBundleMeta(metafile: Metafile): BundleMeta {
+ const bundleMeta = metafile as BundleMeta;
+ Object.entries(bundleMeta.inputs).forEach(([filename, moduleInfo]) => {
+ moduleInfo.imports.forEach(({ path }) => {
+ const moduleInfo = bundleMeta.inputs[path];
+
+ // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition -- can be `undefined`.
+ if (!moduleInfo) {
+ logger.debug(`'${path}' is not exist in esbuild metafile`);
+ return;
+ }
+
+ if (!bundleMeta.inputs[path].parents) {
+ bundleMeta.inputs[path].parents = new Set();
+ }
+
+ // eslint-disable-next-line @typescript-eslint/no-non-null-assertion -- after if statement.
+ bundleMeta.inputs[path].parents!.add(filename);
+ });
+ });
+ return bundleMeta;
+ }
+
+ public async getDelta(
+ modulePath: string,
+ metafile: Metafile,
+ ): Promise {
+ try {
+ performance.mark(`hmr:build:${this.context.id}`);
+ const { code, target, dependencies } = await this.transformRuntime(
+ modulePath,
+ metafile,
+ );
+ const { duration } = performance.measure(
+ 'hmr:build-duration',
+ `hmr:build:${this.context.id}`,
+ );
+ this.stripRoot(modulePath);
+ logger.info(
+ [
+ target,
+ colors.gray(`+ ${dependencies} module(s)`),
+ 'transformed in',
+ colors.cyan(`${Math.floor(duration)}ms`),
+ ].join(' '),
+ );
+ return {
+ id: '',
+ code: this.asFallbackBoundary(code),
+ path: modulePath,
+ fullyReload: false,
+ };
+ } catch (error) {
+ logger.error('unable to transform runtime modules', error as Error);
+ return {
+ id: '',
+ code: '',
+ path: modulePath,
+ fullyReload: true,
+ };
+ }
+ }
+
+ private stripRoot(path: string): string {
+ return path.replace(this.stripRootRegex, '');
+ }
+
+ private async transformRuntime(
+ modulePath: string,
+ bundleMeta: BundleMeta,
+ ): Promise<{ code: string; target: string; dependencies: number }> {
+ const strippedModulePath = this.stripRoot(modulePath);
+ const reverseDependencies = this.getReverseDependencies(
+ strippedModulePath,
+ bundleMeta,
+ );
+ const transformFlags: Record = {};
+ const transformedCodes = await Promise.all(
+ [modulePath, ...reverseDependencies].map(async (modulePath) => {
+ // To avoid re-transformation.
+ if (transformFlags[modulePath]) {
+ return '';
+ }
+ transformFlags[modulePath] = true;
+
+ const rawCode = await fs.readFile(modulePath, 'utf-8');
+ const importPaths = this.getActualImportPaths(
+ this.stripRoot(modulePath),
+ bundleMeta,
+ );
+
+ const { code } = await this.pipeline.transform(
+ rawCode,
+ { ...DUMMY_ESBUILD_ARGS, path: modulePath },
+ { externalPattern: this.context.externalPattern, importPaths },
+ );
+
+ return this.asCallHmrUpdate(modulePath, code);
+ }),
+ );
+
+ return {
+ code: transformedCodes.join('\n'),
+ target: strippedModulePath,
+ dependencies: Object.keys(transformFlags).length,
+ };
+ }
+
+ private getTargetModuleInfo(
+ targetModule: string,
+ bundleMeta: BundleMeta,
+ ): ModuleInfo {
+ // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition -- can be `undefined`
+ if (!bundleMeta.inputs[targetModule]) {
+ throw new Error(`[HMR] ${targetModule} not found in bundle meta`);
+ }
+ return bundleMeta.inputs[targetModule];
+ }
+
+ private getActualImportPaths(
+ targetModule: string,
+ bundleMeta: BundleMeta,
+ ): Record {
+ logger.debug(`${colors.cyan(targetModule)} imports`);
+ const moduleInfo = this.getTargetModuleInfo(targetModule, bundleMeta);
+ const importPaths = moduleInfo.imports.reduce(
+ (prev, curr) => {
+ // To avoid wrong assets path.
+ // eg. `react-native-esbuild-assets:/path/to/assets`
+ const splitted = path.resolve(this.context.root, curr.path).split(':');
+ const actualPath = splitted[splitted.length - 1];
+ if (curr.original && !prev[curr.original]) {
+ logger.debug(
+ `${colors.gray(`├─ ${this.stripRoot(curr.original)} ▸`)} ${
+ this.externalPatternRegex.test(curr.original)
+ ? colors.gray('')
+ : this.stripRoot(actualPath)
+ }`,
+ );
+ }
+ return {
+ ...prev,
+ ...(curr.original ? { [curr.original]: actualPath } : null),
+ };
+ },
+ {} as Record,
+ );
+
+ logger.debug(
+ colors.gray(`╰─ ${Object.keys(importPaths).length} import(s)`),
+ );
+
+ return importPaths;
+ }
+
+ private getReverseDependencies(
+ targetModule: string,
+ bundleMeta: BundleMeta,
+ dependencies: string[] = [],
+ depth = 0,
+ ): string[] {
+ if (depth === 0) logger.debug(colors.gray(targetModule));
+
+ // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition -- can be `undefined`
+ if (bundleMeta.inputs[targetModule]?.parents) {
+ // eslint-disable-next-line @typescript-eslint/no-non-null-assertion -- after if statement.
+ bundleMeta.inputs[targetModule].parents!.forEach((parentModule) => {
+ const spaces = new Array(depth * 3).fill(' ').join('');
+ logger.debug(colors.gray(`${spaces}╰─ ${parentModule}`));
+
+ dependencies = this.getReverseDependencies(
+ parentModule,
+ bundleMeta,
+ [...dependencies, path.join(this.context.root, parentModule)],
+ depth + 1,
+ );
+ });
+ }
+
+ return dependencies;
+ }
+
+ private asCallHmrUpdate(id: string, code: string): string {
+ return `${HMR_UPDATE_FUNCTION}(${JSON.stringify(id)}, function () {
+ ${code}
+ });`;
+ }
+
+ private asFallbackBoundary(code: string): string {
+ return `try {
+ ${code}
+ } catch (error) {
+ console.error('[HMR] unable to accept', error);
+ ${getReloadByDevSettingsProxy()}
+ }`;
+ }
+}
diff --git a/packages/hmr/lib/constants.ts b/packages/hmr/lib/constants.ts
new file mode 100644
index 00000000..93d203a5
--- /dev/null
+++ b/packages/hmr/lib/constants.ts
@@ -0,0 +1,11 @@
+/**
+ * WARNING: Property and function identifiers must match the names defined in `types.ts`.
+ */
+export const HMR_REGISTER_FUNCTION = 'global.__hmr.register';
+export const HMR_UPDATE_FUNCTION = 'global.__hmr.update';
+export const REACT_REFRESH_REGISTER_FUNCTION =
+ 'global.__hmr.reactRefresh.register';
+export const REACT_REFRESH_SIGNATURE_FUNCTION =
+ 'global.__hmr.reactRefresh.getSignature';
+export const PERFORM_REACT_REFRESH_SCRIPT =
+ 'global.__hmr.reactRefresh.performReactRefresh()';
diff --git a/packages/hmr/lib/index.ts b/packages/hmr/lib/index.ts
new file mode 100644
index 00000000..e1f8201e
--- /dev/null
+++ b/packages/hmr/lib/index.ts
@@ -0,0 +1,5 @@
+export { HmrTransformer } from './HmrTransformer';
+export * from './server';
+export * from './constants';
+export type * from './types';
+export type { HmrServer } from './server/HmrServer';
diff --git a/packages/hmr/lib/runtime/setup.ts b/packages/hmr/lib/runtime/setup.ts
new file mode 100644
index 00000000..3a8f3cdb
--- /dev/null
+++ b/packages/hmr/lib/runtime/setup.ts
@@ -0,0 +1,125 @@
+import * as RefreshRuntime from 'react-refresh/runtime';
+
+type HotModuleReplacementAcceptCallback = (payload: {
+ id: HotModuleReplacementId;
+}) => void;
+
+type HotModuleReplacementDisposeCallback = () => void;
+
+if (__DEV__ && typeof global.__hmr === 'undefined') {
+ const HMR_DEBOUNCE_DELAY = 50;
+ let performReactRefreshTimeout: NodeJS.Timeout | null = null;
+
+ class HotModuleReplacementContext {
+ public static registry: Record<
+ HotModuleReplacementId,
+ HotModuleReplacementContext | undefined
+ > = {};
+ public locked = false;
+ public acceptCallbacks: HotModuleReplacementAcceptCallback[] = [];
+ public disposeCallbacks: HotModuleReplacementDisposeCallback[] = [];
+
+ constructor(public id: HotModuleReplacementId) {}
+
+ accept(acceptCallback: HotModuleReplacementAcceptCallback): void {
+ if (this.locked) return;
+ this.acceptCallbacks.push(acceptCallback);
+ }
+
+ dispose(disposeCallback: HotModuleReplacementDisposeCallback): void {
+ if (this.locked) return;
+ this.disposeCallbacks.push(disposeCallback);
+ }
+
+ lock(): void {
+ this.locked = true;
+ }
+ }
+
+ const isReactRefreshBoundary = (type: unknown): boolean => {
+ return Boolean(
+ RefreshRuntime.isLikelyComponentType(type) &&
+ // @ts-expect-error - expect a ReactElement
+ !type?.prototype?.isReactComponent,
+ );
+ };
+
+ const HotModuleReplacementRuntimeModule: HotModuleReplacementRuntimeModule = {
+ register: (id: HotModuleReplacementId) => {
+ const context = HotModuleReplacementContext.registry[id];
+ if (context) {
+ context.lock();
+ return context;
+ }
+ return (HotModuleReplacementContext.registry[id] =
+ new HotModuleReplacementContext(id));
+ },
+ update: (id: HotModuleReplacementId, evalUpdates: () => void) => {
+ const context = HotModuleReplacementContext.registry[id];
+ context?.disposeCallbacks.forEach((callback) => {
+ callback();
+ });
+ evalUpdates();
+ context?.acceptCallbacks.forEach((callback) => {
+ callback({ id });
+ });
+ },
+ reactRefresh: {
+ register: (type: unknown, id: string) => {
+ if (!isReactRefreshBoundary(type)) return;
+ RefreshRuntime.register(type, id);
+ },
+ getSignature: () => {
+ const signature: any =
+ // eslint-disable-next-line @typescript-eslint/no-confusing-void-expression -- Wrong type definition.
+ RefreshRuntime.createSignatureFunctionForTransform();
+ return (
+ type: unknown,
+ id: string,
+ forceReset?: boolean,
+ getCustomHooks?: () => unknown[],
+ ) => {
+ if (!isReactRefreshBoundary(type)) return;
+
+ return signature(type, id, forceReset, getCustomHooks) as unknown;
+ };
+ },
+ performReactRefresh: () => {
+ if (performReactRefreshTimeout !== null) {
+ return;
+ }
+
+ performReactRefreshTimeout = setTimeout(() => {
+ performReactRefreshTimeout = null;
+ if (RefreshRuntime.hasUnrecoverableErrors()) {
+ console.error('[HMR::react-refresh] has unrecoverable errors');
+ return;
+ }
+ RefreshRuntime.performReactRefresh();
+ }, HMR_DEBOUNCE_DELAY);
+ },
+ },
+ };
+
+ RefreshRuntime.injectIntoGlobalHook(global);
+
+ Object.defineProperty(global, '__hmr', {
+ enumerable: false,
+ value: HotModuleReplacementRuntimeModule,
+ });
+
+ // for web
+ // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition -- allow
+ const isWeb = global?.navigator?.appName === 'Netscape';
+ if (isWeb) {
+ const socketURL = new URL('hot', `ws://${window.location.host}`);
+ const socket = new window.WebSocket(socketURL.href);
+ socket.addEventListener('message', (event) => {
+ const payload = window.JSON.parse(event.data);
+ if (payload.type === 'update') {
+ const code = payload.body?.added[0]?.module?.[1];
+ code && window.eval(code);
+ }
+ });
+ }
+}
diff --git a/packages/hmr/lib/server/HmrAppServer.ts b/packages/hmr/lib/server/HmrAppServer.ts
new file mode 100644
index 00000000..0ae88b50
--- /dev/null
+++ b/packages/hmr/lib/server/HmrAppServer.ts
@@ -0,0 +1,54 @@
+import type { Server, MessageEvent, Data } from 'ws';
+import type { HmrClientMessage, HmrMessage, HmrMessageType } from '../types';
+import { HmrServer } from './HmrServer';
+
+export class HmrAppServer extends HmrServer {
+ private messageHandler?: (message: HmrClientMessage) => void;
+
+ constructor() {
+ super();
+ this.setup((socket) => {
+ socket.onmessage = this.handleMessage.bind(this);
+ });
+ }
+
+ private parseClientMessage(data: Data): HmrClientMessage | null {
+ try {
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment -- from ws data
+ const parsedData = JSON.parse(String(data));
+ return 'type' in parsedData ? (parsedData as HmrClientMessage) : null;
+ } catch (error) {
+ return null;
+ }
+ }
+
+ private handleMessage(event: MessageEvent): void {
+ const message = this.parseClientMessage(event.data);
+ if (!message) return;
+
+ /**
+ * @see {@link https://github.com/facebook/metro/blob/v0.77.0/packages/metro/src/HmrServer.js#L200-L239}
+ */
+ this.messageHandler?.(message);
+ }
+
+ public getWebSocketServer(): Server {
+ return this.server;
+ }
+
+ public setMessageHandler(handler: (message: HmrClientMessage) => void): void {
+ this.messageHandler = handler;
+ }
+
+ public send(
+ type: MessageType,
+ body: HmrMessage[MessageType],
+ ): void {
+ this.connectedSocket?.send(
+ JSON.stringify({
+ type,
+ body,
+ }),
+ );
+ }
+}
diff --git a/packages/hmr/lib/server/HmrServer.ts b/packages/hmr/lib/server/HmrServer.ts
new file mode 100644
index 00000000..904b3160
--- /dev/null
+++ b/packages/hmr/lib/server/HmrServer.ts
@@ -0,0 +1,40 @@
+import { Server, type WebSocket } from 'ws';
+import { logger } from '../shared';
+import type { HmrMessage, HmrMessageType } from '../types';
+
+export abstract class HmrServer {
+ protected server: Server;
+ protected connectedSocket?: WebSocket;
+
+ constructor() {
+ this.server = new Server({ noServer: true });
+ }
+
+ private handleClose(): void {
+ this.connectedSocket = undefined;
+ logger.debug('HMR web socket was closed');
+ }
+
+ public setup(onConnect?: (socket: WebSocket) => void): void {
+ this.server.on('connection', (socket) => {
+ this.connectedSocket = socket;
+ this.connectedSocket.onclose = this.handleClose.bind(this);
+ this.connectedSocket.onerror = this.handleClose.bind(this);
+ logger.debug('HMR web socket was connected');
+ onConnect?.(socket);
+ });
+
+ this.server.on('error', (error) => {
+ logger.error('HMR web socket server error', error);
+ });
+ }
+
+ public getWebSocketServer(): Server {
+ return this.server;
+ }
+
+ public abstract send(
+ type: MessageType,
+ body: HmrMessage[MessageType],
+ ): void;
+}
diff --git a/packages/hmr/lib/server/HmrWebServer.ts b/packages/hmr/lib/server/HmrWebServer.ts
new file mode 100644
index 00000000..6c95b075
--- /dev/null
+++ b/packages/hmr/lib/server/HmrWebServer.ts
@@ -0,0 +1,21 @@
+import type { HmrMessage, HmrMessageType } from '../types';
+import { HmrServer } from './HmrServer';
+
+export class HmrWebServer extends HmrServer {
+ constructor() {
+ super();
+ this.setup();
+ }
+
+ public send(
+ type: MessageType,
+ body: HmrMessage[MessageType],
+ ): void {
+ this.connectedSocket?.send(
+ JSON.stringify({
+ type,
+ body,
+ }),
+ );
+ }
+}
diff --git a/packages/hmr/lib/server/index.ts b/packages/hmr/lib/server/index.ts
new file mode 100644
index 00000000..7360b435
--- /dev/null
+++ b/packages/hmr/lib/server/index.ts
@@ -0,0 +1,2 @@
+export * from './HmrAppServer';
+export * from './HmrWebServer';
diff --git a/packages/hmr/lib/shared.ts b/packages/hmr/lib/shared.ts
new file mode 100644
index 00000000..8523c002
--- /dev/null
+++ b/packages/hmr/lib/shared.ts
@@ -0,0 +1,3 @@
+import { Logger } from '@react-native-esbuild/utils';
+
+export const logger = new Logger('hmr');
diff --git a/packages/hmr/lib/types.ts b/packages/hmr/lib/types.ts
new file mode 100644
index 00000000..5ee0cb8c
--- /dev/null
+++ b/packages/hmr/lib/types.ts
@@ -0,0 +1,128 @@
+import type {
+ BabelTransformRule,
+ SwcTransformRule,
+} from '@react-native-esbuild/transformer';
+import type { Metafile, ImportKind } from 'esbuild';
+
+/* eslint-disable no-var -- allow */
+declare global {
+ type HotModuleReplacementId = string;
+
+ interface HotModuleReplacementRuntimeModule {
+ register: (id: HotModuleReplacementId) => void;
+ update: (id: HotModuleReplacementId, evalUpdates: () => void) => void;
+ // react-refresh/runtime
+ reactRefresh: {
+ register: (type: unknown, id: string) => void;
+ getSignature: () => (
+ type: unknown,
+ id: string,
+ forceReset?: boolean,
+ getCustomHooks?: () => unknown[],
+ ) => unknown;
+ performReactRefresh: () => void;
+ };
+ }
+
+ // react-native
+ var __DEV__: boolean;
+
+ // react-refresh/runtime
+ var __hmr: HotModuleReplacementRuntimeModule | undefined;
+
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any -- allow
+ var window: any;
+}
+
+export interface PipelineBuilderOptions {
+ fullyTransformPackageNames?: string[];
+ stripFlowPackageNames?: string[];
+ additionalBabelRules?: BabelTransformRule[];
+ additionalSwcRules?: SwcTransformRule[];
+}
+
+/**
+ * HMR web socket messages
+ * @see {@link https://github.com/facebook/metro/blob/v0.77.0/packages/metro-runtime/src/modules/types.flow.js#L68}
+ */
+export type HmrClientMessage =
+ | RegisterEntryPointsMessage
+ | LogMessage
+ | LogOptInMessage;
+
+export interface RegisterEntryPointsMessage {
+ type: 'register-entrypoints';
+ entryPoints: string[];
+}
+
+export interface LogMessage {
+ type: 'log';
+ level:
+ | 'trace'
+ | 'info'
+ | 'warn'
+ | 'log'
+ | 'group'
+ | 'groupCollapsed'
+ | 'groupEnd'
+ | 'debug';
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any -- follow metro types
+ data: any[];
+ mode: 'BRIDGE' | 'NOBRIDGE';
+}
+
+export interface LogOptInMessage {
+ type: 'log-opt-in';
+}
+
+/**
+ * HMR update message
+ * @see {@link https://github.com/facebook/metro/blob/v0.77.0/packages/metro-runtime/src/modules/types.flow.js#L44-L56}
+ */
+// eslint-disable-next-line @typescript-eslint/consistent-type-definitions -- allow type
+export type HmrMessage = {
+ update: HmrUpdate;
+ 'update-start': {
+ isInitialUpdate: boolean;
+ };
+ 'update-done': undefined;
+};
+
+export type HmrMessageType = keyof HmrMessage;
+
+export interface HmrUpdate {
+ readonly added: HmrModule[];
+ readonly deleted: number[];
+ readonly modified: HmrModule[];
+ isInitialUpdate: boolean;
+ revisionId: string;
+}
+export interface HmrModule {
+ module: [number, string];
+ sourceMappingURL: string | null;
+ sourceURL: string | null;
+}
+
+export type BundleMeta = Metafile & {
+ inputs: Metafile['inputs'] & Record;
+};
+
+// Extended type from `esbuild.Metafile['inputs'][file]`
+export interface ModuleInfo {
+ bytes: number;
+ imports: {
+ path: string;
+ kind: ImportKind;
+ external?: boolean;
+ original?: string;
+ }[];
+ parents?: Set;
+ format?: 'cjs' | 'esm';
+}
+
+export interface BundleUpdate {
+ id: string;
+ path: string;
+ code: string;
+ fullyReload: boolean;
+}
diff --git a/packages/hmr/package.json b/packages/hmr/package.json
new file mode 100644
index 00000000..33e0b98d
--- /dev/null
+++ b/packages/hmr/package.json
@@ -0,0 +1,57 @@
+{
+ "name": "@react-native-esbuild/hmr",
+ "version": "0.1.0-beta.8",
+ "description": "HMR implementation for @react-native-esbuild",
+ "keywords": [
+ "react-native",
+ "esbuild",
+ "hmr",
+ "react-refresh"
+ ],
+ "author": "leegeunhyeok ",
+ "homepage": "https://github.com/leegeunhyeok/react-native-esbuild#readme",
+ "license": "MIT",
+ "type": "commonjs",
+ "module": "lib/index.ts",
+ "main": "dist/index.js",
+ "types": "dist/lib/index.d.ts",
+ "exports": {
+ ".": "./dist/index.js",
+ "./runtime": "./dist/runtime.js"
+ },
+ "directories": {
+ "lib": "lib",
+ "test": "__tests__"
+ },
+ "files": [
+ "lib",
+ "dist"
+ ],
+ "publishConfig": {
+ "access": "public"
+ },
+ "repository": {
+ "type": "git",
+ "url": "git+https://github.com/leegeunhyeok/react-native-esbuild.git"
+ },
+ "scripts": {
+ "prepack": "yarn build",
+ "cleanup": "rimraf ./dist",
+ "build": "node build/index.js && tsc"
+ },
+ "bugs": {
+ "url": "https://github.com/leegeunhyeok/react-native-esbuild/issues"
+ },
+ "devDependencies": {
+ "@swc/helpers": "^0.5.2",
+ "@types/react-refresh": "^0.14.3",
+ "esbuild": "^0.19.3"
+ },
+ "dependencies": {
+ "@react-native-esbuild/internal": "workspace:^",
+ "@react-native-esbuild/transformer": "workspace:^",
+ "@react-native-esbuild/utils": "workspace:*",
+ "react-refresh": "^0.14.0",
+ "ws": "^8.14.2"
+ }
+}
diff --git a/packages/hmr/tsconfig.json b/packages/hmr/tsconfig.json
new file mode 100644
index 00000000..bd42e358
--- /dev/null
+++ b/packages/hmr/tsconfig.json
@@ -0,0 +1,11 @@
+{
+ "extends": "../../tsconfig.json",
+ "compilerOptions": {
+ "rootDir": ".",
+ "outDir": "./dist",
+ "declaration": true,
+ "emitDeclarationOnly": true
+ },
+ "include": ["./lib"],
+ "exclude": ["./dist"]
+}
diff --git a/packages/internal/lib/__tests__/__snapshots__/presets.test.ts.snap b/packages/internal/lib/__tests__/__snapshots__/presets.test.ts.snap
index ddf3b952..618b9261 100644
--- a/packages/internal/lib/__tests__/__snapshots__/presets.test.ts.snap
+++ b/packages/internal/lib/__tests__/__snapshots__/presets.test.ts.snap
@@ -1,7 +1,3 @@
// Jest Snapshot v1, https://goo.gl/fbAQLP
-exports[`getAssetRegistrationScript should match snapshot 1`] = `
-"
- module.exports = require('react-native/Libraries/Image/AssetRegistry').registerAsset({"__packager_asset":true,"name":"image","type":"png","scales":[1,2,3],"hash":"hash","httpServerLocation":"/image.png","width":0,"height":0});
- "
-`;
+exports[`getAssetRegistrationScript should match snapshot 1`] = `"module.exports =require('react-native/Libraries/Image/AssetRegistry').registerAsset({"__packager_asset":true,"name":"image","type":"png","scales":[1,2,3],"hash":"hash","httpServerLocation":"/image.png","width":0,"height":0});"`;
diff --git a/packages/internal/lib/presets.ts b/packages/internal/lib/presets.ts
index 5cdd1b05..aa6d11e9 100644
--- a/packages/internal/lib/presets.ts
+++ b/packages/internal/lib/presets.ts
@@ -22,19 +22,13 @@ export const getInjectVariables = (dev: boolean): string[] => [
`global = typeof globalThis !== 'undefined' ? globalThis : typeof global !== 'undefined' ? global : typeof window !== 'undefined' ? window : this`,
];
-const getReactNativePolyfills = (root: string): Promise => {
+const getReactNativePolyfills = (root: string): string[] => {
// eslint-disable-next-line @typescript-eslint/no-var-requires -- Allow dynamic require.
const getPolyfills = require(
resolveFromRoot(REACT_NATIVE_GET_POLYFILLS_PATH, root),
) as () => string[];
- return Promise.all(
- getPolyfills().map((scriptPath) =>
- fs
- .readFile(scriptPath, { encoding: 'utf-8' })
- .then((code) => wrapWithIIFE(code, scriptPath)),
- ),
- );
+ return getPolyfills();
};
export const getReactNativeInitializeCore = (root: string): string => {
@@ -56,16 +50,24 @@ export const getPreludeScript = async (
{ dev = true }: BundleOptions,
root: string,
): Promise => {
- const polyfills = await getReactNativePolyfills(root);
+ const scripts = await Promise.all(
+ getReactNativePolyfills(root).map((scriptPath) =>
+ fs
+ .readFile(scriptPath, { encoding: 'utf-8' })
+ .then((code) => wrapWithIIFE(code, scriptPath)),
+ ),
+ );
+
const initialScripts = [
`var ${getInjectVariables(dev).join(',')};`,
`process.env=process.env||{};`,
`process.env.NODE_ENV=${JSON.stringify(getNodeEnv(dev))};`,
- ...polyfills,
+ ...scripts,
].join('\n');
return initialScripts;
};
+
/**
* Get asset registration script.
*
@@ -83,20 +85,18 @@ export const getAssetRegistrationScript = ({
Asset,
'name' | 'type' | 'scales' | 'hash' | 'httpServerLocation' | 'dimensions'
>): string => {
- return `
- module.exports = require('react-native/Libraries/Image/AssetRegistry').registerAsset(${JSON.stringify(
- {
- __packager_asset: true,
- name,
- type,
- scales,
- hash,
- httpServerLocation,
- width: dimensions.width,
- height: dimensions.height,
- },
- )});
- `;
+ return `module.exports =require('react-native/Libraries/Image/AssetRegistry').registerAsset(${JSON.stringify(
+ {
+ __packager_asset: true,
+ name,
+ type,
+ scales,
+ hash,
+ httpServerLocation,
+ width: dimensions.width,
+ height: dimensions.height,
+ },
+ )});`;
};
/**
diff --git a/packages/plugins/lib/assetRegisterPlugin/assetRegisterPlugin.ts b/packages/plugins/lib/assetRegisterPlugin/assetRegisterPlugin.ts
index b115bfda..ccb86f10 100644
--- a/packages/plugins/lib/assetRegisterPlugin/assetRegisterPlugin.ts
+++ b/packages/plugins/lib/assetRegisterPlugin/assetRegisterPlugin.ts
@@ -6,6 +6,7 @@ import {
type Asset,
} from '@react-native-esbuild/internal';
import { ASSET_EXTENSIONS } from '@react-native-esbuild/internal';
+import { HmrTransformer } from '@react-native-esbuild/hmr';
import type { AssetRegisterPluginConfig, SuffixPathResult } from '../types';
import {
copyAssetsToDestination,
@@ -96,12 +97,19 @@ export const createAssetRegisterPlugin: ReactNativeEsbuildPluginCreator<
build.onLoad({ filter: /./, namespace: ASSET_NAMESPACE }, async (args) => {
const asset = await resolveScaledAssets(context, args);
+ const assetRegistrationScript = getAssetRegistrationScript(asset);
assets.push(asset);
return {
resolveDir: path.dirname(args.path),
- contents: getAssetRegistrationScript(asset),
+ contents: context.enableHmr
+ ? HmrTransformer.registerAsExternalModule(
+ args.path,
+ assetRegistrationScript,
+ 'module.exports',
+ )
+ : assetRegistrationScript,
loader: 'js',
};
});
diff --git a/packages/plugins/lib/reactNativeRuntimeTransformPlugin/helpers/index.ts b/packages/plugins/lib/reactNativeRuntimeTransformPlugin/helpers/index.ts
index c9314409..d4fdc5e5 100644
--- a/packages/plugins/lib/reactNativeRuntimeTransformPlugin/helpers/index.ts
+++ b/packages/plugins/lib/reactNativeRuntimeTransformPlugin/helpers/index.ts
@@ -1 +1,2 @@
export * from './caches';
+export * from './json';
diff --git a/packages/plugins/lib/reactNativeRuntimeTransformPlugin/helpers/json.ts b/packages/plugins/lib/reactNativeRuntimeTransformPlugin/helpers/json.ts
new file mode 100644
index 00000000..755febe4
--- /dev/null
+++ b/packages/plugins/lib/reactNativeRuntimeTransformPlugin/helpers/json.ts
@@ -0,0 +1,31 @@
+import fs from 'node:fs/promises';
+import type { PluginBuild } from 'esbuild';
+import { HmrTransformer } from '@react-native-esbuild/hmr';
+
+/**
+ * When development mode & HMR enabled, the '.json' contents
+ * must be registered in the global module registry for HMR.
+ */
+export const jsonAsJsModule = (build: PluginBuild): void => {
+ build.onLoad({ filter: /\.json$/ }, async (args) => {
+ const rawJson = await fs.readFile(args.path, { encoding: 'utf-8' });
+ const parsedJson = JSON.parse(rawJson) as Record;
+ const identifier = 'json';
+
+ return {
+ contents: HmrTransformer.registerAsExternalModule(
+ args.path,
+ `const ${identifier} = ${rawJson};
+ ${Object.keys(parsedJson)
+ .map((member) => {
+ const memberName = JSON.stringify(member);
+ return `exports[${memberName}] = ${identifier}[${memberName}]`;
+ })
+ .join('\n')}
+ module.exports = ${identifier};`,
+ identifier,
+ ),
+ loader: 'js',
+ };
+ });
+};
diff --git a/packages/plugins/lib/reactNativeRuntimeTransformPlugin/reactNativeRuntimeTransformPlugin.ts b/packages/plugins/lib/reactNativeRuntimeTransformPlugin/reactNativeRuntimeTransformPlugin.ts
index 6b9d3026..e3a9990d 100644
--- a/packages/plugins/lib/reactNativeRuntimeTransformPlugin/reactNativeRuntimeTransformPlugin.ts
+++ b/packages/plugins/lib/reactNativeRuntimeTransformPlugin/reactNativeRuntimeTransformPlugin.ts
@@ -1,6 +1,5 @@
import fs from 'node:fs/promises';
import path from 'node:path';
-import type { OnLoadResult } from 'esbuild';
import {
ReactNativeEsbuildBundler as Bundler,
type ReactNativeEsbuildPluginCreator,
@@ -11,6 +10,11 @@ import {
swcPresets,
type AsyncTransformStep,
} from '@react-native-esbuild/transformer';
+import {
+ HmrTransformer,
+ REACT_REFRESH_REGISTER_FUNCTION,
+ REACT_REFRESH_SIGNATURE_FUNCTION,
+} from '@react-native-esbuild/hmr';
import { logger } from '../shared';
import type { ReactNativeRuntimeTransformPluginConfig } from '../types';
import {
@@ -18,6 +22,7 @@ import {
getTransformedCodeFromFileSystemCache,
writeTransformedCodeToInMemoryCache,
writeTransformedCodeToFileSystemCache,
+ jsonAsJsModule,
} from './helpers';
const NAME = 'react-native-runtime-transform-plugin';
@@ -28,7 +33,6 @@ export const createReactNativeRuntimeTransformPlugin: ReactNativeEsbuildPluginCr
name: NAME,
setup: (build): void => {
const cacheController = Bundler.caches.get(context.id);
- const bundlerSharedData = Bundler.shared.get(context.id);
const cacheEnabled = context.config.cache ?? true;
const {
stripFlowPackageNames = [],
@@ -38,33 +42,38 @@ export const createReactNativeRuntimeTransformPlugin: ReactNativeEsbuildPluginCr
const additionalBabelRules = additionalTransformRules?.babel ?? [];
const additionalSwcRules = additionalTransformRules?.swc ?? [];
const injectScriptPaths = [
- getReactNativeInitializeCore(context.root),
+ ...[
+ getReactNativeInitializeCore(context.root),
+ // `hmr/runtime` should import after `initializeCore` initialized.
+ context.enableHmr ? '@react-native-esbuild/hmr/runtime' : undefined,
+ ],
...(config?.injectScriptPaths ?? []),
- ];
-
- const reactNativeRuntimePreset = swcPresets.getReactNativeRuntimePreset(
- context.config.transformer?.jsc,
- );
+ ].filter(Boolean) as string[];
+
+ const reactNativeRuntimePreset = swcPresets.getReactNativeRuntimePreset({
+ experimental: {
+ hmr: context.enableHmr
+ ? {
+ runtime: false,
+ refreshReg: REACT_REFRESH_REGISTER_FUNCTION,
+ refreshSig: REACT_REFRESH_SIGNATURE_FUNCTION,
+ }
+ : undefined,
+ },
+ });
const onBeforeTransform: AsyncTransformStep = async (
code,
- args,
+ _args,
moduleMeta,
) => {
- const isChangedFile = bundlerSharedData.watcher.changed === args.path;
const cacheConfig = {
hash: moduleMeta.hash,
mtimeMs: moduleMeta.stats.mtimeMs,
};
- // 1. Force re-transform when file is changed.
- if (isChangedFile) {
- logger.debug('changed file detected', { path: args.path });
- return { code, done: false };
- }
-
/**
- * 2. Use previous transformed result and skip transform
+ * 1. Use previous transformed result and skip transform
* when file is not changed and transform result exist in memory.
*/
const inMemoryCache = getTransformedCodeFromInMemoryCache(
@@ -75,12 +84,12 @@ export const createReactNativeRuntimeTransformPlugin: ReactNativeEsbuildPluginCr
return { code: inMemoryCache, done: true };
}
- // 3. Transform code on each build task when cache is disabled.
+ // 2. Transform code on each build task when cache is disabled.
if (!cacheEnabled) {
return { code, done: false };
}
- // 4. Trying to get cache from file system.
+ // 3. Trying to get cache from file system.
// = cache exist ? use cache : transform code
const cachedCode = await getTransformedCodeFromFileSystemCache(
cacheController,
@@ -92,7 +101,7 @@ export const createReactNativeRuntimeTransformPlugin: ReactNativeEsbuildPluginCr
const onAfterTransform: AsyncTransformStep = async (
code,
- _args,
+ args,
moduleMeta,
) => {
const cacheConfig = {
@@ -109,11 +118,16 @@ export const createReactNativeRuntimeTransformPlugin: ReactNativeEsbuildPluginCr
);
}
- return { code, done: true };
+ return {
+ code:
+ context.enableHmr && HmrTransformer.isBoundary(args.path)
+ ? HmrTransformer.asBoundary(args.path, code)
+ : code,
+ done: true,
+ };
};
- let transformPipeline: AsyncTransformPipeline;
- const transformPipelineBuilder = new AsyncTransformPipeline.builder(context)
+ const transformPipeline = new AsyncTransformPipeline.builder(context)
.setSwcPreset(reactNativeRuntimePreset)
.setInjectScripts(injectScriptPaths)
.setFullyTransformPackages(fullyTransformPackageNames)
@@ -121,18 +135,21 @@ export const createReactNativeRuntimeTransformPlugin: ReactNativeEsbuildPluginCr
.setAdditionalBabelTransformRules(additionalBabelRules)
.setAdditionalSwcTransformRules(additionalSwcRules)
.onStart(onBeforeTransform)
- .onEnd(onAfterTransform);
+ .onEnd(onAfterTransform)
+ .build();
- build.onStart(() => {
- transformPipeline = transformPipelineBuilder.build();
- });
+ context.enableHmr && jsonAsJsModule(build);
build.onLoad({ filter: /\.(?:[mc]js|[tj]sx?)$/ }, async (args) => {
const rawCode = await fs.readFile(args.path, { encoding: 'utf-8' });
return {
- contents: (await transformPipeline.transform(rawCode, args)).code,
+ contents: (
+ await transformPipeline.transform(rawCode, args, {
+ externalPattern: context.externalPattern,
+ })
+ ).code,
loader: 'js',
- } as OnLoadResult;
+ };
});
build.onEnd(async (args) => {
diff --git a/packages/plugins/lib/svgTransformPlugin/svgTransformPlugin.ts b/packages/plugins/lib/svgTransformPlugin/svgTransformPlugin.ts
index 41005e78..8ebb001e 100644
--- a/packages/plugins/lib/svgTransformPlugin/svgTransformPlugin.ts
+++ b/packages/plugins/lib/svgTransformPlugin/svgTransformPlugin.ts
@@ -1,6 +1,8 @@
import fs from 'node:fs/promises';
import { transform } from '@svgr/core';
import type { ReactNativeEsbuildPluginCreator } from '@react-native-esbuild/core';
+import { HmrTransformer } from '@react-native-esbuild/hmr';
+import { defaultTemplate, SVG_COMPONENT_NAME } from './templates';
const NAME = 'svg-transform-plugin';
@@ -13,15 +15,24 @@ export const createSvgTransformPlugin: ReactNativeEsbuildPluginCreator = (
build.onLoad({ filter: /\.svg$/ }, async (args) => {
const rawSvg = await fs.readFile(args.path, { encoding: 'utf8' });
+ const svgTransformedCode = await transform(
+ rawSvg,
+ {
+ template: defaultTemplate,
+ plugins: ['@svgr/plugin-jsx'],
+ native: isNative,
+ },
+ { filePath: args.path },
+ );
+
return {
- contents: await transform(
- rawSvg,
- {
- plugins: ['@svgr/plugin-jsx'],
- native: isNative,
- },
- { filePath: args.path },
- ),
+ contents: context.enableHmr
+ ? HmrTransformer.registerAsExternalModule(
+ args.path,
+ svgTransformedCode,
+ SVG_COMPONENT_NAME,
+ )
+ : svgTransformedCode,
loader: 'jsx',
};
});
diff --git a/packages/plugins/lib/svgTransformPlugin/templates/index.ts b/packages/plugins/lib/svgTransformPlugin/templates/index.ts
new file mode 100644
index 00000000..3fd30d45
--- /dev/null
+++ b/packages/plugins/lib/svgTransformPlugin/templates/index.ts
@@ -0,0 +1,16 @@
+import type { Config } from '@svgr/core';
+
+export const SVG_COMPONENT_NAME = 'SvgLogo';
+
+export const defaultTemplate: Config['template'] = (variables, { tpl }) => {
+ return tpl`
+${variables.imports};
+
+${variables.interfaces};
+
+const ${SVG_COMPONENT_NAME} = (${variables.props}) => (
+ ${variables.jsx}
+);
+
+${variables.exports};`;
+};
diff --git a/packages/plugins/package.json b/packages/plugins/package.json
index aac9060c..946fc2a5 100644
--- a/packages/plugins/package.json
+++ b/packages/plugins/package.json
@@ -46,6 +46,7 @@
"dependencies": {
"@react-native-esbuild/config": "workspace:*",
"@react-native-esbuild/core": "workspace:*",
+ "@react-native-esbuild/hmr": "workspace:*",
"@react-native-esbuild/internal": "workspace:*",
"@react-native-esbuild/transformer": "workspace:*",
"@react-native-esbuild/utils": "workspace:*",
diff --git a/packages/transformer/README.md b/packages/transformer/README.md
index 2982f484..29ac1ee5 100644
--- a/packages/transformer/README.md
+++ b/packages/transformer/README.md
@@ -7,11 +7,9 @@ import {
stripFlowWithSucrase,
transformWithBabel,
transformWithSwc,
- minifyWithSwc,
} from '@react-native-esbuild/transformer';
await stripFlowWithSucrase(code, context);
await transformWithBabel(code, context, options);
await transformWithSwc(code, context, options);
-await minifyWithSwc(code, context, options);
```
diff --git a/packages/transformer/lib/helpers/transformer.ts b/packages/transformer/lib/helpers/transformer.ts
index 4e4877b3..daf06066 100644
--- a/packages/transformer/lib/helpers/transformer.ts
+++ b/packages/transformer/lib/helpers/transformer.ts
@@ -27,7 +27,10 @@ export const transformByBabelRule = (
context: TransformerContext,
): Promise => {
return rule.test(context.path, code)
- ? transformWithBabel(code, context, ruleOptionsToPreset(rule.options, code))
+ ? transformWithBabel(code, {
+ context,
+ preset: ruleOptionsToPreset(rule.options, code),
+ })
: Promise.resolve(null);
};
@@ -37,11 +40,10 @@ export const transformSyncByBabelRule = (
context: TransformerContext,
): string | null => {
return rule.test(context.path, code)
- ? transformSyncWithBabel(
- code,
+ ? transformSyncWithBabel(code, {
context,
- ruleOptionsToPreset(rule.options, code),
- )
+ preset: ruleOptionsToPreset(rule.options, code),
+ })
: null;
};
@@ -51,7 +53,10 @@ export const transformBySwcRule = (
context: TransformerContext,
): Promise => {
return rule.test(context.path, code)
- ? transformWithSwc(code, context, ruleOptionsToPreset(rule.options, code))
+ ? transformWithSwc(code, {
+ context,
+ preset: ruleOptionsToPreset(rule.options, code),
+ })
: Promise.resolve(null);
};
@@ -61,10 +66,9 @@ export const transformSyncBySwcRule = (
context: TransformerContext,
): string | null => {
return rule.test(context.path, code)
- ? transformSyncWithSwc(
- code,
+ ? transformSyncWithSwc(code, {
context,
- ruleOptionsToPreset(rule.options, code),
- )
+ preset: ruleOptionsToPreset(rule.options, code),
+ })
: null;
};
diff --git a/packages/transformer/lib/pipelines/AsyncTransformPipeline.ts b/packages/transformer/lib/pipelines/AsyncTransformPipeline.ts
index e3330a0c..f9002b99 100644
--- a/packages/transformer/lib/pipelines/AsyncTransformPipeline.ts
+++ b/packages/transformer/lib/pipelines/AsyncTransformPipeline.ts
@@ -8,8 +8,8 @@ import {
} from '../transformer';
import { transformByBabelRule, transformBySwcRule } from '../helpers';
import type { AsyncTransformStep, ModuleMeta, TransformResult } from '../types';
-import { TransformPipeline } from './pipeline';
-import { TransformPipelineBuilder } from './builder';
+import { TransformPipeline } from './TransformPipeline';
+import { TransformPipelineBuilder } from './TransformPipelineBuilder';
export class AsyncTransformPipelineBuilder extends TransformPipelineBuilder<
AsyncTransformStep,
@@ -40,14 +40,14 @@ export class AsyncTransformPipelineBuilder extends TransformPipelineBuilder<
this.fullyTransformPackageNames,
);
if (fullyTransformPackagesRegExp) {
- pipeline.addStep(async (code, args) => {
+ pipeline.addStep(async (code, args, moduleMeta) => {
if (fullyTransformPackagesRegExp.test(args.path)) {
return {
- code: await transformWithBabel(
- code,
- this.getContext(args),
- this.presets.babelFullyTransform,
- ),
+ code: await transformWithBabel(code, {
+ moduleMeta,
+ context: this.getContext(args),
+ preset: this.presets.babelFullyTransform,
+ }),
// skip other transformations when fully transformed
done: true,
};
@@ -61,12 +61,15 @@ export class AsyncTransformPipelineBuilder extends TransformPipelineBuilder<
this.stripFlowPackageNames,
);
if (stripFlowPackageNamesRegExp) {
- pipeline.addStep((code, args) => {
+ pipeline.addStep((code, args, moduleMeta) => {
if (
stripFlowPackageNamesRegExp.test(args.path) ||
this.isFlow(code, args.path)
) {
- code = stripFlowWithSucrase(code, this.getContext(args));
+ code = stripFlowWithSucrase(code, {
+ moduleMeta,
+ context: this.getContext(args),
+ });
}
return Promise.resolve({ code, done: false });
@@ -96,13 +99,13 @@ export class AsyncTransformPipelineBuilder extends TransformPipelineBuilder<
}
// 6. Transform code to es5.
- pipeline.addStep(async (code, args) => {
+ pipeline.addStep(async (code, args, moduleMeta) => {
return {
- code: await transformWithSwc(
- code,
- this.getContext(args),
- this.swcPreset,
- ),
+ code: await transformWithSwc(code, {
+ moduleMeta,
+ context: this.getContext(args),
+ preset: this.swcPreset,
+ }),
done: true,
};
});
@@ -117,9 +120,14 @@ export class AsyncTransformPipeline extends TransformPipeline {
+ async transform(
+ code: string,
+ args: OnLoadArgs,
+ baseModuleMeta?: Pick,
+ ): Promise {
const fileStat = await fs.stat(args.path);
const moduleMeta: ModuleMeta = {
+ ...baseModuleMeta,
stats: fileStat,
hash: this.getHash(this.context.id, args.path, fileStat.mtimeMs),
};
diff --git a/packages/transformer/lib/pipelines/SyncTransformPipeline.ts b/packages/transformer/lib/pipelines/SyncTransformPipeline.ts
index 8eef2f51..edae191c 100644
--- a/packages/transformer/lib/pipelines/SyncTransformPipeline.ts
+++ b/packages/transformer/lib/pipelines/SyncTransformPipeline.ts
@@ -8,8 +8,8 @@ import {
} from '../transformer';
import { transformSyncByBabelRule, transformSyncBySwcRule } from '../helpers';
import type { SyncTransformStep, TransformResult, ModuleMeta } from '../types';
-import { TransformPipeline } from './pipeline';
-import { TransformPipelineBuilder } from './builder';
+import { TransformPipeline } from './TransformPipeline';
+import { TransformPipelineBuilder } from './TransformPipelineBuilder';
export class SyncTransformPipelineBuilder extends TransformPipelineBuilder<
SyncTransformStep,
@@ -40,14 +40,14 @@ export class SyncTransformPipelineBuilder extends TransformPipelineBuilder<
this.fullyTransformPackageNames,
);
if (fullyTransformPackagesRegExp) {
- pipeline.addStep((code, args) => {
+ pipeline.addStep((code, args, moduleMeta) => {
if (fullyTransformPackagesRegExp.test(args.path)) {
return {
- code: transformSyncWithBabel(
- code,
- this.getContext(args),
- this.presets.babelFullyTransform,
- ),
+ code: transformSyncWithBabel(code, {
+ moduleMeta,
+ context: this.getContext(args),
+ preset: this.presets.babelFullyTransform,
+ }),
// skip other transformations when fully transformed
done: true,
};
@@ -61,12 +61,15 @@ export class SyncTransformPipelineBuilder extends TransformPipelineBuilder<
this.stripFlowPackageNames,
);
if (stripFlowPackageNamesRegExp) {
- pipeline.addStep((code, args) => {
+ pipeline.addStep((code, args, moduleMeta) => {
if (
stripFlowPackageNamesRegExp.test(args.path) ||
this.isFlow(code, args.path)
) {
- code = stripFlowWithSucrase(code, this.getContext(args));
+ code = stripFlowWithSucrase(code, {
+ moduleMeta,
+ context: this.getContext(args),
+ });
}
return { code, done: false };
@@ -96,9 +99,13 @@ export class SyncTransformPipelineBuilder extends TransformPipelineBuilder<
}
// 6. Transform code to es5.
- pipeline.addStep((code, args) => {
+ pipeline.addStep((code, args, moduleMeta) => {
return {
- code: transformSyncWithSwc(code, this.getContext(args), this.swcPreset),
+ code: transformSyncWithSwc(code, {
+ moduleMeta,
+ context: this.getContext(args),
+ preset: this.swcPreset,
+ }),
done: true,
};
});
@@ -128,9 +135,14 @@ export class SyncTransformPipeline extends TransformPipeline
return this;
}
- transform(code: string, args: OnLoadArgs): TransformResult {
+ transform(
+ code: string,
+ args: OnLoadArgs,
+ baseModuleMeta?: Pick,
+ ): TransformResult {
const fileStat = fs.statSync(args.path);
const moduleMeta: ModuleMeta = {
+ ...baseModuleMeta,
stats: fileStat,
hash: this.getHash(this.context.id, args.path, fileStat.mtimeMs),
};
diff --git a/packages/transformer/lib/pipelines/pipeline.ts b/packages/transformer/lib/pipelines/TransformPipeline.ts
similarity index 82%
rename from packages/transformer/lib/pipelines/pipeline.ts
rename to packages/transformer/lib/pipelines/TransformPipeline.ts
index 84c84c3e..034ddec1 100644
--- a/packages/transformer/lib/pipelines/pipeline.ts
+++ b/packages/transformer/lib/pipelines/TransformPipeline.ts
@@ -1,6 +1,6 @@
import type { OnLoadArgs } from 'esbuild';
import md5 from 'md5';
-import type { TransformStep, TransformerContext } from '../types';
+import type { ModuleMeta, TransformStep, TransformerContext } from '../types';
export abstract class TransformPipeline> {
protected steps: Step[] = [];
@@ -36,5 +36,9 @@ export abstract class TransformPipeline> {
return this;
}
- abstract transform(code: string, args: OnLoadArgs): ReturnType;
+ abstract transform(
+ code: string,
+ args: OnLoadArgs,
+ baseModuleMeta?: Pick,
+ ): ReturnType;
}
diff --git a/packages/transformer/lib/pipelines/builder.ts b/packages/transformer/lib/pipelines/TransformPipelineBuilder.ts
similarity index 97%
rename from packages/transformer/lib/pipelines/builder.ts
rename to packages/transformer/lib/pipelines/TransformPipelineBuilder.ts
index 366f9e28..c4374971 100644
--- a/packages/transformer/lib/pipelines/builder.ts
+++ b/packages/transformer/lib/pipelines/TransformPipelineBuilder.ts
@@ -8,7 +8,7 @@ import type {
TransformerOptionsPreset,
} from '../types';
import { babelPresets } from '../transformer';
-import type { TransformPipeline } from './pipeline';
+import type { TransformPipeline } from './TransformPipeline';
const FLOW_SYMBOL = ['@flow', '@noflow'] as const;
diff --git a/packages/transformer/lib/transformer/babel/babel.ts b/packages/transformer/lib/transformer/babel/babel.ts
index 95261d9c..a03c13ed 100644
--- a/packages/transformer/lib/transformer/babel/babel.ts
+++ b/packages/transformer/lib/transformer/babel/babel.ts
@@ -19,10 +19,9 @@ const loadBabelOptions = (
export const transformWithBabel: AsyncTransformer = async (
code: string,
- context,
- preset,
+ { context, moduleMeta, preset },
) => {
- const babelOptions = loadBabelOptions(context, preset?.(context));
+ const babelOptions = loadBabelOptions(context, preset?.(context, moduleMeta));
if (!babelOptions) {
throw new Error('cannot load babel options');
}
@@ -37,10 +36,9 @@ export const transformWithBabel: AsyncTransformer = async (
export const transformSyncWithBabel: SyncTransformer = (
code: string,
- context,
- preset,
+ { context, moduleMeta, preset },
) => {
- const babelOptions = loadBabelOptions(context, preset?.(context));
+ const babelOptions = loadBabelOptions(context, preset?.(context, moduleMeta));
if (!babelOptions) {
throw new Error('cannot load babel options');
}
diff --git a/packages/transformer/lib/transformer/sucrase/sucrase.ts b/packages/transformer/lib/transformer/sucrase/sucrase.ts
index a0d5baf2..5fc752ee 100644
--- a/packages/transformer/lib/transformer/sucrase/sucrase.ts
+++ b/packages/transformer/lib/transformer/sucrase/sucrase.ts
@@ -10,7 +10,10 @@ const stripFlowTypeofImportStatements = (code: string): string => {
.join('\n');
};
-export const stripFlowWithSucrase: SyncTransformer = (code, context) => {
+export const stripFlowWithSucrase: SyncTransformer = (
+ code,
+ { context },
+) => {
return stripFlowTypeofImportStatements(
transform(code, {
transforms: TRANSFORM_FOR_STRIP_FLOW,
diff --git a/packages/transformer/lib/transformer/swc/presets.ts b/packages/transformer/lib/transformer/swc/presets.ts
index 62eb43f2..8978c00f 100644
--- a/packages/transformer/lib/transformer/swc/presets.ts
+++ b/packages/transformer/lib/transformer/swc/presets.ts
@@ -6,9 +6,22 @@ import type {
} from '@swc/core';
import type {
TransformerOptionsPreset,
+ ReactNativeRuntimePresetOptions,
SwcJestPresetOptions,
+ ModuleMeta,
+ SwcMinifyPresetOptions,
+ TransformerContext,
} from '../../types';
+/**
+ * TODO: move into hmr package.
+ *
+ * @see `HmrTransformer.isBoundary`
+ */
+const isHMRBoundary = (path: string): boolean => {
+ return !path.includes('/node_modules/') && !path.endsWith('runtime.js');
+};
+
const getParserOptions = (path: string): TsParserConfig | EsParserConfig => {
return /\.tsx?$/.test(path)
? ({
@@ -23,13 +36,35 @@ const getParserOptions = (path: string): TsParserConfig | EsParserConfig => {
} as EsParserConfig);
};
+const getSwcExperimental = (
+ context: TransformerContext,
+ moduleMeta?: ModuleMeta,
+ options?: ReactNativeRuntimePresetOptions,
+): JscConfig['experimental'] => {
+ if (options?.experimental?.hmr && isHMRBoundary(context.path)) {
+ return {
+ plugins: [
+ [
+ 'swc-plugin-global-module',
+ {
+ runtimeModule: options.experimental.hmr.runtime,
+ externalPattern: moduleMeta?.externalPattern,
+ importPaths: moduleMeta?.importPaths,
+ },
+ ],
+ ],
+ };
+ }
+ return undefined;
+};
+
/**
* swc transform options preset for react-native runtime.
*/
const getReactNativeRuntimePreset = (
- jscConfig?: Pick,
+ options?: ReactNativeRuntimePresetOptions,
): TransformerOptionsPreset => {
- return (context) => ({
+ return (context, moduleMeta) => ({
minify: false,
sourceMaps: false,
isModule: true,
@@ -39,10 +74,22 @@ const getReactNativeRuntimePreset = (
parser: getParserOptions(context.path),
target: 'es5',
loose: false,
- externalHelpers: true,
+ externalHelpers: !context.dev,
keepClassNames: true,
- transform: jscConfig?.transform,
- experimental: jscConfig?.experimental,
+ transform: {
+ react: {
+ development: context.dev,
+ // @ts-expect-error -- wrong type definition.
+ refresh:
+ options?.experimental?.hmr && isHMRBoundary(context.path)
+ ? {
+ refreshReg: options.experimental.hmr.refreshReg,
+ refreshSig: options.experimental.hmr.refreshSig,
+ }
+ : undefined,
+ },
+ },
+ experimental: getSwcExperimental(context, moduleMeta, options),
},
filename: context.path,
root: context.root,
@@ -95,11 +142,21 @@ const getJestPreset = (
});
};
-const getMinifyPreset = () => {
- return () => ({
- compress: true,
- mangle: true,
- sourceMap: false,
+const getMinifyPreset = ({
+ minify,
+}: SwcMinifyPresetOptions): TransformerOptionsPreset => {
+ return (context) => ({
+ minify,
+ inputSourceMap: false,
+ inlineSourcesContent: false,
+ jsc: {
+ parser: getParserOptions(context.path),
+ target: 'es5',
+ loose: false,
+ keepClassNames: true,
+ },
+ filename: context.path,
+ root: context.root,
});
};
diff --git a/packages/transformer/lib/transformer/swc/swc.ts b/packages/transformer/lib/transformer/swc/swc.ts
index 9fe79f90..eb512f8c 100644
--- a/packages/transformer/lib/transformer/swc/swc.ts
+++ b/packages/transformer/lib/transformer/swc/swc.ts
@@ -1,18 +1,14 @@
-import {
- transform,
- transformSync,
- minify,
- type Options,
- type JsMinifyOptions,
-} from '@swc/core';
+import { transform, transformSync, type Options } from '@swc/core';
import type { AsyncTransformer, SyncTransformer } from '../../types';
export const transformWithSwc: AsyncTransformer = async (
code,
- context,
- preset,
+ { context, moduleMeta, preset },
) => {
- const { code: transformedCode } = await transform(code, preset?.(context));
+ const { code: transformedCode } = await transform(
+ code,
+ preset?.(context, moduleMeta),
+ );
if (typeof transformedCode !== 'string') {
throw new Error('swc transformed source is empty');
@@ -23,10 +19,12 @@ export const transformWithSwc: AsyncTransformer = async (
export const transformSyncWithSwc: SyncTransformer = (
code,
- context,
- preset,
+ { context, moduleMeta, preset },
) => {
- const { code: transformedCode } = transformSync(code, preset?.(context));
+ const { code: transformedCode } = transformSync(
+ code,
+ preset?.(context, moduleMeta),
+ );
if (typeof transformedCode !== 'string') {
throw new Error('swc transformed source is empty');
@@ -34,17 +32,3 @@ export const transformSyncWithSwc: SyncTransformer = (
return transformedCode;
};
-
-export const minifyWithSwc: AsyncTransformer = async (
- code,
- context,
- preset,
-) => {
- const { code: minifiedCode } = await minify(code, preset?.(context));
-
- if (typeof minifiedCode !== 'string') {
- throw new Error('swc minified source is empty');
- }
-
- return minifiedCode;
-};
diff --git a/packages/transformer/lib/types.ts b/packages/transformer/lib/types.ts
index 52591bbe..6ea2dc90 100644
--- a/packages/transformer/lib/types.ts
+++ b/packages/transformer/lib/types.ts
@@ -5,16 +5,20 @@ import type { Options as SwcTransformOptions } from '@swc/core';
export type AsyncTransformer = (
code: string,
- context: TransformerContext,
- preset?: TransformerOptionsPreset,
+ config: TransformerConfig,
) => Promise;
export type SyncTransformer = (
code: string,
- context: TransformerContext,
- preset?: TransformerOptionsPreset,
+ config: TransformerConfig,
) => string;
+interface TransformerConfig {
+ context: TransformerContext;
+ moduleMeta?: ModuleMeta;
+ preset?: TransformerOptionsPreset;
+}
+
export interface TransformerContext {
id: number;
root: string;
@@ -25,11 +29,30 @@ export interface TransformerContext {
export type TransformerOptionsPreset = (
context: TransformerContext,
+ moduleMeta?: ModuleMeta,
) => TransformerOptions;
-// swc preset options
-export interface SwcReactNativeRuntimePresetOptions {
- reactRefresh?: { moduleId: string };
+export interface ReactNativeRuntimePresetOptions {
+ /**
+ * Options for experimental features.
+ */
+ experimental?: {
+ /**
+ * HMR(Hot Module Replacement) options.
+ *
+ * If `undefined`, HMR will be disabled.
+ */
+ hmr?: {
+ /**
+ * `runtimeModule` option in `swc-plugin-global-module`.
+ *
+ * @see github {@link https://github.com/leegeunhyeok/swc-plugin-global-module}
+ */
+ runtime: boolean;
+ refreshReg: string;
+ refreshSig: string;
+ };
+ };
}
export interface SwcJestPresetOptions {
@@ -63,6 +86,10 @@ export interface SwcJestPresetOptions {
};
}
+export interface SwcMinifyPresetOptions {
+ minify: boolean;
+}
+
export interface TransformRuleBase {
/**
* Predicator for transform
@@ -95,4 +122,6 @@ export interface TransformResult {
export interface ModuleMeta {
hash: string;
stats: Stats;
+ externalPattern?: string;
+ importPaths?: Record;
}
diff --git a/packages/transformer/package.json b/packages/transformer/package.json
index 4c10fcb3..ecbed7df 100644
--- a/packages/transformer/package.json
+++ b/packages/transformer/package.json
@@ -49,6 +49,7 @@
"md5": "^2.3.0",
"sucrase": "^3.34.0",
"swc-plugin-coverage-instrument": "^0.0.20",
+ "swc-plugin-global-module": "^0.1.0-alpha.5",
"swc_mut_cjs_exports": "^0.85.0"
}
}
diff --git a/shared/index.js b/shared/index.js
index 931f6fc5..33c3386d 100644
--- a/shared/index.js
+++ b/shared/index.js
@@ -1,15 +1,24 @@
const path = require('node:path');
/**
- * @param {string} entryFile filename
+ * @param {string} packageDir build script directory
* @param {import('esbuild').BuildOptions} options additional options
* @returns {import('esbuild').BuildOptions}
*/
-exports.getEsbuildBaseOptions = (packageDir, options = {}) => ({
- entryPoints: [path.resolve(packageDir, '../lib/index.ts')],
+const getEsbuildBaseOptions = (packageDir, options = {}) => ({
+ entryPoints: [path.join(getPackageRoot(packageDir), 'lib/index.ts')],
outfile: 'dist/index.js',
bundle: true,
platform: 'node',
packages: 'external',
...options,
});
+
+/**
+ * @param {string} packageDir build script directory
+ * @returns package root path
+ */
+const getPackageRoot = (packageDir) => path.resolve(packageDir, '../');
+
+exports.getEsbuildBaseOptions = getEsbuildBaseOptions;
+exports.getPackageRoot = getPackageRoot;
diff --git a/yarn.lock b/yarn.lock
index 42602e43..7b219581 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -3650,6 +3650,7 @@ __metadata:
"@babel/core": ^7.23.2
"@faker-js/faker": ^8.1.0
"@react-native-esbuild/config": "workspace:*"
+ "@react-native-esbuild/hmr": "workspace:*"
"@react-native-esbuild/internal": "workspace:*"
"@react-native-esbuild/transformer": "workspace:*"
"@react-native-esbuild/utils": "workspace:*"
@@ -3661,6 +3662,7 @@ __metadata:
ora: ^5.4.1
peerDependencies:
react-native: "*"
+ swc-plugin-global-module: "*"
languageName: unknown
linkType: soft
@@ -3672,6 +3674,7 @@ __metadata:
"@react-native-community/cli-server-api": ^11.3.6
"@react-native-esbuild/config": "workspace:*"
"@react-native-esbuild/core": "workspace:*"
+ "@react-native-esbuild/hmr": "workspace:*"
"@react-native-esbuild/internal": "workspace:*"
"@react-native-esbuild/symbolicate": "workspace:*"
"@react-native-esbuild/utils": "workspace:*"
@@ -3702,7 +3705,22 @@ __metadata:
languageName: unknown
linkType: soft
-"@react-native-esbuild/internal@workspace:*, @react-native-esbuild/internal@workspace:packages/internal":
+"@react-native-esbuild/hmr@workspace:*, @react-native-esbuild/hmr@workspace:packages/hmr":
+ version: 0.0.0-use.local
+ resolution: "@react-native-esbuild/hmr@workspace:packages/hmr"
+ dependencies:
+ "@react-native-esbuild/internal": "workspace:^"
+ "@react-native-esbuild/transformer": "workspace:^"
+ "@react-native-esbuild/utils": "workspace:*"
+ "@swc/helpers": ^0.5.2
+ "@types/react-refresh": ^0.14.3
+ esbuild: ^0.19.3
+ react-refresh: ^0.14.0
+ ws: ^8.14.2
+ languageName: unknown
+ linkType: soft
+
+"@react-native-esbuild/internal@workspace:*, @react-native-esbuild/internal@workspace:^, @react-native-esbuild/internal@workspace:packages/internal":
version: 0.0.0-use.local
resolution: "@react-native-esbuild/internal@workspace:packages/internal"
dependencies:
@@ -3736,6 +3754,7 @@ __metadata:
"@faker-js/faker": ^8.1.0
"@react-native-esbuild/config": "workspace:*"
"@react-native-esbuild/core": "workspace:*"
+ "@react-native-esbuild/hmr": "workspace:*"
"@react-native-esbuild/internal": "workspace:*"
"@react-native-esbuild/transformer": "workspace:*"
"@react-native-esbuild/utils": "workspace:*"
@@ -3761,7 +3780,7 @@ __metadata:
languageName: unknown
linkType: soft
-"@react-native-esbuild/transformer@workspace:*, @react-native-esbuild/transformer@workspace:packages/transformer":
+"@react-native-esbuild/transformer@workspace:*, @react-native-esbuild/transformer@workspace:^, @react-native-esbuild/transformer@workspace:packages/transformer":
version: 0.0.0-use.local
resolution: "@react-native-esbuild/transformer@workspace:packages/transformer"
dependencies:
@@ -3774,6 +3793,7 @@ __metadata:
md5: ^2.3.0
sucrase: ^3.34.0
swc-plugin-coverage-instrument: ^0.0.20
+ swc-plugin-global-module: ^0.1.0-alpha.5
swc_mut_cjs_exports: ^0.85.0
languageName: unknown
linkType: soft
@@ -4414,7 +4434,7 @@ __metadata:
languageName: node
linkType: hard
-"@swc/helpers@npm:^0.5.3":
+"@swc/helpers@npm:^0.5.2, @swc/helpers@npm:^0.5.3":
version: 0.5.3
resolution: "@swc/helpers@npm:0.5.3"
dependencies:
@@ -4541,7 +4561,7 @@ __metadata:
languageName: node
linkType: hard
-"@types/babel__core@npm:^7.1.14":
+"@types/babel__core@npm:*, @types/babel__core@npm:^7.1.14":
version: 7.20.3
resolution: "@types/babel__core@npm:7.20.3"
dependencies:
@@ -4840,6 +4860,16 @@ __metadata:
languageName: node
linkType: hard
+"@types/react-refresh@npm:^0.14.3":
+ version: 0.14.3
+ resolution: "@types/react-refresh@npm:0.14.3"
+ dependencies:
+ "@types/babel__core": "*"
+ csstype: ^3.0.2
+ checksum: ff48c18f928f213cfd1fa379f6b275e401adad87f909a83501a7026075a1d596ef2697269a9ce4fc5063841622ea2c2ffe9c332a2cc9e5ca67199f05c7994f89
+ languageName: node
+ linkType: hard
+
"@types/react-test-renderer@npm:^18.0.5":
version: 18.0.5
resolution: "@types/react-test-renderer@npm:18.0.5"
@@ -8201,7 +8231,7 @@ __metadata:
languageName: node
linkType: hard
-"esbuild@npm:^0.19.5":
+"esbuild@npm:^0.19.3, esbuild@npm:^0.19.5":
version: 0.19.5
resolution: "esbuild@npm:0.19.5"
dependencies:
@@ -17694,6 +17724,13 @@ __metadata:
languageName: node
linkType: hard
+"swc-plugin-global-module@npm:^0.1.0-alpha.5":
+ version: 0.1.0-alpha.5
+ resolution: "swc-plugin-global-module@npm:0.1.0-alpha.5"
+ checksum: 10523c54644544a989bdd77cf45a273f529d191816d8c5a4cb992267845048d66125e720a71bca49ae2ab47fcd051077f024d00cf4cf31030eeac1c67082ac18
+ languageName: node
+ linkType: hard
+
"swc_mut_cjs_exports@npm:^0.85.0":
version: 0.85.0
resolution: "swc_mut_cjs_exports@npm:0.85.0"
@@ -19122,7 +19159,7 @@ __metadata:
languageName: node
linkType: hard
-"ws@npm:^8.13.0":
+"ws@npm:^8.13.0, ws@npm:^8.14.2":
version: 8.14.2
resolution: "ws@npm:8.14.2"
peerDependencies: