metro 0.84.2 → 0.84.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (42) hide show
  1. package/package.json +15 -16
  2. package/src/Assets.js +20 -13
  3. package/src/Assets.js.flow +20 -13
  4. package/src/Bundler/util.js.flow +3 -3
  5. package/src/DeltaBundler/DeltaCalculator.d.ts +1 -11
  6. package/src/DeltaBundler/DeltaCalculator.js +55 -46
  7. package/src/DeltaBundler/DeltaCalculator.js.flow +72 -61
  8. package/src/DeltaBundler/getTransformCacheKey.js +3 -1
  9. package/src/DeltaBundler/getTransformCacheKey.js.flow +7 -2
  10. package/src/HmrServer.d.ts +15 -3
  11. package/src/HmrServer.js +7 -0
  12. package/src/HmrServer.js.flow +15 -5
  13. package/src/ModuleGraph/worker/collectDependencies.js +52 -0
  14. package/src/ModuleGraph/worker/collectDependencies.js.flow +67 -0
  15. package/src/Server.d.ts +4 -1
  16. package/src/Server.js +42 -5
  17. package/src/Server.js.flow +45 -5
  18. package/src/index.d.ts +22 -3
  19. package/src/index.flow.js +2 -2
  20. package/src/index.flow.js.flow +23 -4
  21. package/src/lib/JsonReporter.js.flow +2 -2
  22. package/src/lib/TerminalReporter.js +39 -41
  23. package/src/lib/TerminalReporter.js.flow +51 -32
  24. package/src/lib/getAppendScripts.js.flow +2 -2
  25. package/src/lib/logToConsole.js +8 -7
  26. package/src/lib/logToConsole.js.flow +7 -7
  27. package/src/lib/reporting.js +16 -7
  28. package/src/lib/reporting.js.flow +16 -5
  29. package/src/node-haste/DependencyGraph/ModuleResolution.d.ts +9 -22
  30. package/src/node-haste/DependencyGraph/ModuleResolution.js +4 -22
  31. package/src/node-haste/DependencyGraph/ModuleResolution.js.flow +10 -59
  32. package/src/node-haste/DependencyGraph/createFileMap.js +1 -2
  33. package/src/node-haste/DependencyGraph/createFileMap.js.flow +4 -3
  34. package/src/node-haste/DependencyGraph.d.ts +2 -5
  35. package/src/node-haste/DependencyGraph.js +22 -11
  36. package/src/node-haste/DependencyGraph.js.flow +24 -13
  37. package/src/node-haste/PackageCache.d.ts +12 -16
  38. package/src/node-haste/PackageCache.js +65 -54
  39. package/src/node-haste/PackageCache.js.flow +103 -79
  40. package/src/node-haste/Package.d.ts +0 -28
  41. package/src/node-haste/Package.js +0 -28
  42. package/src/node-haste/Package.js.flow +0 -39
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "metro",
3
- "version": "0.84.2",
3
+ "version": "0.84.4",
4
4
  "description": "🚇 The JavaScript bundler for React Native.",
5
5
  "main": "src/index.js",
6
6
  "bin": "src/cli.js",
@@ -27,31 +27,30 @@
27
27
  "@babel/traverse": "^7.29.0",
28
28
  "@babel/types": "^7.29.0",
29
29
  "accepts": "^2.0.0",
30
- "chalk": "^4.0.0",
31
30
  "ci-info": "^2.0.0",
32
31
  "connect": "^3.6.5",
33
32
  "debug": "^4.4.0",
34
33
  "error-stack-parser": "^2.0.6",
35
34
  "flow-enums-runtime": "^0.0.6",
36
35
  "graceful-fs": "^4.2.4",
37
- "hermes-parser": "0.33.3",
36
+ "hermes-parser": "0.35.0",
38
37
  "image-size": "^1.0.2",
39
38
  "invariant": "^2.2.4",
40
39
  "jest-worker": "^29.7.0",
41
40
  "jsc-safe-url": "^0.2.2",
42
41
  "lodash.throttle": "^4.1.1",
43
- "metro-babel-transformer": "0.84.2",
44
- "metro-cache": "0.84.2",
45
- "metro-cache-key": "0.84.2",
46
- "metro-config": "0.84.2",
47
- "metro-core": "0.84.2",
48
- "metro-file-map": "0.84.2",
49
- "metro-resolver": "0.84.2",
50
- "metro-runtime": "0.84.2",
51
- "metro-source-map": "0.84.2",
52
- "metro-symbolicate": "0.84.2",
53
- "metro-transform-plugins": "0.84.2",
54
- "metro-transform-worker": "0.84.2",
42
+ "metro-babel-transformer": "0.84.4",
43
+ "metro-cache": "0.84.4",
44
+ "metro-cache-key": "0.84.4",
45
+ "metro-config": "0.84.4",
46
+ "metro-core": "0.84.4",
47
+ "metro-file-map": "0.84.4",
48
+ "metro-resolver": "0.84.4",
49
+ "metro-runtime": "0.84.4",
50
+ "metro-source-map": "0.84.4",
51
+ "metro-symbolicate": "0.84.4",
52
+ "metro-transform-plugins": "0.84.4",
53
+ "metro-transform-worker": "0.84.4",
55
54
  "mime-types": "^3.0.1",
56
55
  "nullthrows": "^1.1.1",
57
56
  "serialize-error": "^2.1.0",
@@ -72,7 +71,7 @@
72
71
  "dedent": "^0.7.0",
73
72
  "jest-snapshot": "^29.7.0",
74
73
  "jest-snapshot-serializer-raw": "^1.2.0",
75
- "metro-babel-register": "0.84.2",
74
+ "metro-babel-register": "0.84.4",
76
75
  "metro-memory-fs": "*",
77
76
  "mock-req": "^0.2.0",
78
77
  "mock-res": "^0.6.0",
package/src/Assets.js CHANGED
@@ -221,26 +221,33 @@ async function getAsset(
221
221
  `'${relativePath}' cannot be loaded as its extension is not registered in assetExts`,
222
222
  );
223
223
  }
224
- if (fileExistsInFileMap != null) {
225
- if (!fileExistsInFileMap(absolutePath)) {
226
- throw new Error(
227
- `'${relativePath}' could not be found, because it is not within the projectRoot or watchFolders, or it is blocked via the resolver.blockList config`,
228
- );
229
- }
230
- } else {
231
- if (!pathBelongsToRoots(absolutePath, [projectRoot, ...watchFolders])) {
232
- throw new Error(
233
- `'${relativePath}' could not be found, because it cannot be found in the project root or any watch folder`,
234
- );
235
- }
224
+ if (
225
+ fileExistsInFileMap == null &&
226
+ !pathBelongsToRoots(absolutePath, [projectRoot, ...watchFolders])
227
+ ) {
228
+ throw new Error(
229
+ `'${relativePath}' could not be found, because it cannot be found in the project root or any watch folder`,
230
+ );
236
231
  }
237
232
  const record = await getAbsoluteAssetRecord(absolutePath, platform ?? null);
238
233
  for (let i = 0; i < record.scales.length; i++) {
239
234
  if (record.scales[i] >= assetData.resolution) {
235
+ if (
236
+ fileExistsInFileMap != null &&
237
+ !fileExistsInFileMap(record.files[i])
238
+ ) {
239
+ continue;
240
+ }
240
241
  return _fs.default.promises.readFile(record.files[i]);
241
242
  }
242
243
  }
243
- return _fs.default.promises.readFile(record.files[record.files.length - 1]);
244
+ const lastFile = record.files[record.files.length - 1];
245
+ if (fileExistsInFileMap != null && !fileExistsInFileMap(lastFile)) {
246
+ throw new Error(
247
+ `'${relativePath}' could not be found, because it is not within the projectRoot or watchFolders, or it is blocked via the resolver.blockList config`,
248
+ );
249
+ }
250
+ return _fs.default.promises.readFile(lastFile);
244
251
  }
245
252
  function pathBelongsToRoots(pathToCheck, roots) {
246
253
  for (const rootFolder of roots) {
@@ -299,29 +299,36 @@ export async function getAsset(
299
299
  }
300
300
 
301
301
  // NOTE: If fileExistsInFileMap is not provided, we fall back to pathBelongsToRoots for backward compatibility, as getAsset is part of the public API.
302
- if (fileExistsInFileMap != null) {
303
- if (!fileExistsInFileMap(absolutePath)) {
304
- throw new Error(
305
- `'${relativePath}' could not be found, because it is not within the projectRoot or watchFolders, or it is blocked via the resolver.blockList config`,
306
- );
307
- }
308
- } else {
309
- if (!pathBelongsToRoots(absolutePath, [projectRoot, ...watchFolders])) {
310
- throw new Error(
311
- `'${relativePath}' could not be found, because it cannot be found in the project root or any watch folder`,
312
- );
313
- }
302
+ if (
303
+ fileExistsInFileMap == null &&
304
+ !pathBelongsToRoots(absolutePath, [projectRoot, ...watchFolders])
305
+ ) {
306
+ throw new Error(
307
+ `'${relativePath}' could not be found, because it cannot be found in the project root or any watch folder`,
308
+ );
314
309
  }
315
310
 
316
311
  const record = await getAbsoluteAssetRecord(absolutePath, platform ?? null);
317
312
 
318
313
  for (let i = 0; i < record.scales.length; i++) {
319
314
  if (record.scales[i] >= assetData.resolution) {
315
+ if (
316
+ fileExistsInFileMap != null &&
317
+ !fileExistsInFileMap(record.files[i])
318
+ ) {
319
+ continue;
320
+ }
320
321
  return fs.promises.readFile(record.files[i]);
321
322
  }
322
323
  }
323
324
 
324
- return fs.promises.readFile(record.files[record.files.length - 1]);
325
+ const lastFile = record.files[record.files.length - 1];
326
+ if (fileExistsInFileMap != null && !fileExistsInFileMap(lastFile)) {
327
+ throw new Error(
328
+ `'${relativePath}' could not be found, because it is not within the projectRoot or watchFolders, or it is blocked via the resolver.blockList config`,
329
+ );
330
+ }
331
+ return fs.promises.readFile(lastFile);
325
332
  }
326
333
 
327
334
  function pathBelongsToRoots(
@@ -17,7 +17,7 @@ import * as babylon from '@babel/parser';
17
17
  import template from '@babel/template';
18
18
  import * as babelTypes from '@babel/types';
19
19
 
20
- type SubTree<T: ModuleTransportLike> = (
20
+ type SubTree<T extends ModuleTransportLike> = (
21
21
  moduleTransport: T,
22
22
  moduleTransportsByPath: Map<string, T>,
23
23
  ) => Iterable<number>;
@@ -66,7 +66,7 @@ function filterObject(
66
66
  return copied;
67
67
  }
68
68
 
69
- export function createRamBundleGroups<T: ModuleTransportLike>(
69
+ export function createRamBundleGroups<T extends ModuleTransportLike>(
70
70
  ramGroups: ReadonlyArray<string>,
71
71
  groupableModules: ReadonlyArray<T>,
72
72
  subtree: SubTree<T>,
@@ -123,7 +123,7 @@ export function createRamBundleGroups<T: ModuleTransportLike>(
123
123
  return result;
124
124
  }
125
125
 
126
- function* filter<A: number, B: number>(
126
+ function* filter<A extends number, B extends number>(
127
127
  iterator: ArrayMap<A, B>,
128
128
  predicate: ([A, Array<B>]) => boolean,
129
129
  ): Generator<[A, Array<B>], void, void> {
@@ -6,7 +6,7 @@
6
6
  *
7
7
  * @noformat
8
8
  * @oncall react_native
9
- * @generated SignedSource<<b825a92cd295988e6e972591f11221fd>>
9
+ * @generated SignedSource<<d06b53dd09157df95aeb941035d4ebf0>>
10
10
  *
11
11
  * This file was translated from Flow by scripts/generateTypeScriptDefinitions.js
12
12
  * Original file: packages/metro/src/DeltaBundler/DeltaCalculator.js
@@ -16,7 +16,6 @@
16
16
  */
17
17
 
18
18
  import type {DeltaResult, Options} from './types';
19
- import type {RootPerfLogger} from 'metro-config';
20
19
  import type {ChangeEvent} from 'metro-file-map';
21
20
 
22
21
  import {Graph} from './Graph';
@@ -60,15 +59,6 @@ declare class DeltaCalculator<T> extends EventEmitter {
60
59
  */
61
60
  getGraph(): Graph<T>;
62
61
  _handleMultipleFileChanges: (changeEvent: ChangeEvent) => void;
63
- /**
64
- * Handles a single file change. To avoid doing any work before it's needed,
65
- * the listener only stores the modified file, which will then be used later
66
- * when the delta needs to be calculated.
67
- */
68
- _handleFileChange: (
69
- $$PARAM_0$$: ChangeEvent['eventsQueue'][number],
70
- logger: null | undefined | RootPerfLogger,
71
- ) => unknown;
72
62
  _getChangedDependencies(
73
63
  modifiedFiles: Set<string>,
74
64
  deletedFiles: Set<string>,
@@ -5,12 +5,14 @@ Object.defineProperty(exports, "__esModule", {
5
5
  });
6
6
  exports.default = void 0;
7
7
  var _Graph = require("./Graph");
8
+ var _crypto = _interopRequireDefault(require("crypto"));
8
9
  var _events = _interopRequireDefault(require("events"));
9
10
  var _path = _interopRequireDefault(require("path"));
10
11
  function _interopRequireDefault(e) {
11
12
  return e && e.__esModule ? e : { default: e };
12
13
  }
13
14
  const debug = require("debug")("Metro:DeltaCalculator");
15
+ const changeEventIds = new WeakMap();
14
16
  class DeltaCalculator extends _events.default {
15
17
  _deletedFiles = new Set();
16
18
  _modifiedFiles = new Set();
@@ -94,60 +96,67 @@ class DeltaCalculator extends _events.default {
94
96
  getGraph() {
95
97
  return this._graph;
96
98
  }
97
- _handleMultipleFileChanges = (changeEvent) => {
98
- changeEvent.eventsQueue.forEach((eventInfo) => {
99
- this._handleFileChange(eventInfo, changeEvent.logger);
100
- });
101
- };
102
- _handleFileChange = ({ type, filePath, metadata }, logger) => {
103
- debug("Handling %s: %s (type: %s)", type, filePath, metadata.type);
99
+ #shouldReset(canonicalPath, metadata) {
100
+ if (metadata.isSymlink) {
101
+ return true;
102
+ }
104
103
  if (
105
- metadata.type === "l" ||
106
- (this._options.unstable_enablePackageExports &&
107
- filePath.endsWith(_path.default.sep + "package.json"))
104
+ this._options.unstable_enablePackageExports &&
105
+ (canonicalPath === "package.json" ||
106
+ canonicalPath.endsWith(_path.default.sep + "package.json"))
108
107
  ) {
109
- this._requiresReset = true;
110
- this.emit("change", {
111
- logger,
112
- });
108
+ return true;
109
+ }
110
+ return false;
111
+ }
112
+ _handleMultipleFileChanges = (changeEvent) => {
113
+ const { changes, logger, rootDir } = changeEvent;
114
+ for (const [canonicalPath, metadata] of changes.addedFiles) {
115
+ debug("Handling add: %s", canonicalPath);
116
+ if (this.#shouldReset(canonicalPath, metadata)) {
117
+ this._requiresReset = true;
118
+ }
119
+ const absolutePath = _path.default.join(rootDir, canonicalPath);
120
+ if (this._deletedFiles.has(absolutePath)) {
121
+ this._deletedFiles.delete(absolutePath);
122
+ this._modifiedFiles.add(absolutePath);
123
+ } else {
124
+ this._addedFiles.add(absolutePath);
125
+ this._modifiedFiles.delete(absolutePath);
126
+ }
113
127
  }
114
- let state;
115
- if (this._deletedFiles.has(filePath)) {
116
- state = "deleted";
117
- } else if (this._modifiedFiles.has(filePath)) {
118
- state = "modified";
119
- } else if (this._addedFiles.has(filePath)) {
120
- state = "added";
128
+ for (const [canonicalPath, metadata] of changes.modifiedFiles) {
129
+ debug("Handling change: %s", canonicalPath);
130
+ if (this.#shouldReset(canonicalPath, metadata)) {
131
+ this._requiresReset = true;
132
+ }
133
+ const absolutePath = _path.default.join(rootDir, canonicalPath);
134
+ if (!this._addedFiles.has(absolutePath)) {
135
+ this._modifiedFiles.add(absolutePath);
136
+ }
137
+ this._deletedFiles.delete(absolutePath);
121
138
  }
122
- let nextState;
123
- if (type === "delete") {
124
- nextState = "deleted";
125
- } else if (type === "add") {
126
- nextState = state === "deleted" ? "modified" : "added";
127
- } else {
128
- nextState = state === "added" ? "added" : "modified";
139
+ for (const [canonicalPath, metadata] of changes.removedFiles) {
140
+ debug("Handling delete: %s", canonicalPath);
141
+ if (this.#shouldReset(canonicalPath, metadata)) {
142
+ this._requiresReset = true;
143
+ }
144
+ const absolutePath = _path.default.resolve(rootDir, canonicalPath);
145
+ if (this._addedFiles.has(absolutePath)) {
146
+ this._addedFiles.delete(absolutePath);
147
+ } else {
148
+ this._deletedFiles.add(absolutePath);
149
+ this._modifiedFiles.delete(absolutePath);
150
+ }
129
151
  }
130
- switch (nextState) {
131
- case "deleted":
132
- this._deletedFiles.add(filePath);
133
- this._modifiedFiles.delete(filePath);
134
- this._addedFiles.delete(filePath);
135
- break;
136
- case "added":
137
- this._addedFiles.add(filePath);
138
- this._deletedFiles.delete(filePath);
139
- this._modifiedFiles.delete(filePath);
140
- break;
141
- case "modified":
142
- this._modifiedFiles.add(filePath);
143
- this._deletedFiles.delete(filePath);
144
- this._addedFiles.delete(filePath);
145
- break;
146
- default:
147
- nextState;
152
+ let changeId = changeEventIds.get(changeEvent);
153
+ if (changeId == null) {
154
+ changeId = _crypto.default.randomUUID();
155
+ changeEventIds.set(changeEvent, changeId);
148
156
  }
149
157
  this.emit("change", {
150
158
  logger,
159
+ changeId,
151
160
  });
152
161
  };
153
162
  async _getChangedDependencies(modifiedFiles, deletedFiles, addedFiles) {
@@ -10,16 +10,24 @@
10
10
  */
11
11
 
12
12
  import type {DeltaResult, Options} from './types';
13
- import type {RootPerfLogger} from 'metro-config';
14
13
  import type {ChangeEvent} from 'metro-file-map';
15
14
 
16
15
  import {Graph} from './Graph';
16
+ import crypto from 'crypto';
17
17
  import EventEmitter from 'events';
18
18
  import path from 'path';
19
19
 
20
20
  // eslint-disable-next-line import/no-commonjs
21
21
  const debug = require('debug')('Metro:DeltaCalculator');
22
22
 
23
+ /**
24
+ * Assigns a unique, stable `changeId` to each `ChangeEvent` from the file
25
+ * watcher. Since all `DeltaCalculator` instances share the same
26
+ * `ChangeEvent` object reference per file system change, the `WeakMap`
27
+ * ensures each gets the same `changeId`.
28
+ */
29
+ const changeEventIds: WeakMap<ChangeEvent, string> = new WeakMap();
30
+
23
31
  /**
24
32
  * This class is in charge of calculating the delta of changed modules that
25
33
  * happen between calls. To do so, it subscribes to file changes, so it can
@@ -172,76 +180,79 @@ export default class DeltaCalculator<T> extends EventEmitter {
172
180
  return this._graph;
173
181
  }
174
182
 
175
- _handleMultipleFileChanges = (changeEvent: ChangeEvent) => {
176
- changeEvent.eventsQueue.forEach(eventInfo => {
177
- this._handleFileChange(eventInfo, changeEvent.logger);
178
- });
179
- };
183
+ #shouldReset(
184
+ canonicalPath: string,
185
+ metadata: {+isSymlink: boolean, ...},
186
+ ): boolean {
187
+ if (metadata.isSymlink) {
188
+ return true;
189
+ }
180
190
 
181
- /**
182
- * Handles a single file change. To avoid doing any work before it's needed,
183
- * the listener only stores the modified file, which will then be used later
184
- * when the delta needs to be calculated.
185
- */
186
- _handleFileChange = (
187
- {type, filePath, metadata}: ChangeEvent['eventsQueue'][number],
188
- logger: ?RootPerfLogger,
189
- ): unknown => {
190
- debug('Handling %s: %s (type: %s)', type, filePath, metadata.type);
191
191
  if (
192
- metadata.type === 'l' ||
193
- (this._options.unstable_enablePackageExports &&
194
- filePath.endsWith(path.sep + 'package.json'))
192
+ this._options.unstable_enablePackageExports &&
193
+ (canonicalPath === 'package.json' ||
194
+ canonicalPath.endsWith(path.sep + 'package.json'))
195
195
  ) {
196
- this._requiresReset = true;
197
- this.emit('change', {logger});
196
+ return true;
198
197
  }
199
- let state: void | 'deleted' | 'modified' | 'added';
200
- if (this._deletedFiles.has(filePath)) {
201
- state = 'deleted';
202
- } else if (this._modifiedFiles.has(filePath)) {
203
- state = 'modified';
204
- } else if (this._addedFiles.has(filePath)) {
205
- state = 'added';
198
+
199
+ return false;
200
+ }
201
+
202
+ _handleMultipleFileChanges = (changeEvent: ChangeEvent) => {
203
+ const {changes, logger, rootDir} = changeEvent;
204
+
205
+ // Process added files: deleted+added = modified, otherwise added
206
+ for (const [canonicalPath, metadata] of changes.addedFiles) {
207
+ debug('Handling add: %s', canonicalPath);
208
+ if (this.#shouldReset(canonicalPath, metadata)) {
209
+ this._requiresReset = true;
210
+ }
211
+ const absolutePath = path.join(rootDir, canonicalPath);
212
+ if (this._deletedFiles.has(absolutePath)) {
213
+ this._deletedFiles.delete(absolutePath);
214
+ this._modifiedFiles.add(absolutePath);
215
+ } else {
216
+ this._addedFiles.add(absolutePath);
217
+ this._modifiedFiles.delete(absolutePath);
218
+ }
206
219
  }
207
220
 
208
- let nextState: 'deleted' | 'modified' | 'added';
209
- if (type === 'delete') {
210
- nextState = 'deleted';
211
- } else if (type === 'add') {
212
- // A deleted+added file is modified
213
- nextState = state === 'deleted' ? 'modified' : 'added';
214
- } else {
215
- // type === 'change'
216
- // An added+modified file is added
217
- nextState = state === 'added' ? 'added' : 'modified';
221
+ // Process modified files: added+modified stays added, otherwise modified
222
+ for (const [canonicalPath, metadata] of changes.modifiedFiles) {
223
+ debug('Handling change: %s', canonicalPath);
224
+ if (this.#shouldReset(canonicalPath, metadata)) {
225
+ this._requiresReset = true;
226
+ }
227
+ const absolutePath = path.join(rootDir, canonicalPath);
228
+ if (!this._addedFiles.has(absolutePath)) {
229
+ this._modifiedFiles.add(absolutePath);
230
+ }
231
+ this._deletedFiles.delete(absolutePath);
218
232
  }
219
233
 
220
- switch (nextState) {
221
- case 'deleted':
222
- this._deletedFiles.add(filePath);
223
- this._modifiedFiles.delete(filePath);
224
- this._addedFiles.delete(filePath);
225
- break;
226
- case 'added':
227
- this._addedFiles.add(filePath);
228
- this._deletedFiles.delete(filePath);
229
- this._modifiedFiles.delete(filePath);
230
- break;
231
- case 'modified':
232
- this._modifiedFiles.add(filePath);
233
- this._deletedFiles.delete(filePath);
234
- this._addedFiles.delete(filePath);
235
- break;
236
- default:
237
- nextState as empty;
234
+ // Process removed files: added+deleted = no change, otherwise deleted
235
+ for (const [canonicalPath, metadata] of changes.removedFiles) {
236
+ debug('Handling delete: %s', canonicalPath);
237
+ if (this.#shouldReset(canonicalPath, metadata)) {
238
+ this._requiresReset = true;
239
+ }
240
+ const absolutePath = path.resolve(rootDir, canonicalPath);
241
+ if (this._addedFiles.has(absolutePath)) {
242
+ this._addedFiles.delete(absolutePath);
243
+ } else {
244
+ this._deletedFiles.add(absolutePath);
245
+ this._modifiedFiles.delete(absolutePath);
246
+ }
238
247
  }
239
248
 
240
- // Notify users that there is a change in some of the bundle files. This
241
- // way the client can choose to refetch the bundle.
242
- this.emit('change', {
243
- logger,
244
- });
249
+ let changeId = changeEventIds.get(changeEvent);
250
+ if (changeId == null) {
251
+ changeId = crypto.randomUUID();
252
+ changeEventIds.set(changeEvent, changeId);
253
+ }
254
+
255
+ this.emit('change', {logger, changeId});
245
256
  };
246
257
 
247
258
  async _getChangedDependencies(
@@ -14,7 +14,9 @@ function getTransformCacheKey(opts) {
14
14
  const { transformerPath, transformerConfig } = opts.transformerConfig;
15
15
  const Transformer = require.call(null, transformerPath);
16
16
  const transformerKey = Transformer.getCacheKey
17
- ? Transformer.getCacheKey(transformerConfig)
17
+ ? Transformer.getCacheKey(transformerConfig, {
18
+ projectRoot: opts.projectRoot,
19
+ })
18
20
  : "";
19
21
  return _crypto.default
20
22
  .createHash("sha1")
@@ -19,7 +19,10 @@ import {getCacheKey} from 'metro-cache-key';
19
19
  const VERSION = require('../../package.json').version;
20
20
 
21
21
  type CacheKeyProvider = {
22
- getCacheKey?: JsTransformerConfig => string,
22
+ getCacheKey?: (
23
+ config: JsTransformerConfig,
24
+ opts?: Readonly<{projectRoot: string}>,
25
+ ) => string,
23
26
  };
24
27
 
25
28
  export default function getTransformCacheKey(opts: {
@@ -32,7 +35,9 @@ export default function getTransformCacheKey(opts: {
32
35
  // eslint-disable-next-line no-useless-call
33
36
  const Transformer: CacheKeyProvider = require.call(null, transformerPath);
34
37
  const transformerKey = Transformer.getCacheKey
35
- ? Transformer.getCacheKey(transformerConfig)
38
+ ? Transformer.getCacheKey(transformerConfig, {
39
+ projectRoot: opts.projectRoot,
40
+ })
36
41
  : '';
37
42
 
38
43
  return crypto
@@ -5,7 +5,7 @@
5
5
  * LICENSE file in the root directory of this source tree.
6
6
  *
7
7
  * @noformat
8
- * @generated SignedSource<<759aad52b112d43c0af68fdad28e4453>>
8
+ * @generated SignedSource<<ab4c245134631e14db114a9d49da79d1>>
9
9
  *
10
10
  * This file was translated from Flow by scripts/generateTypeScriptDefinitions.js
11
11
  * Original file: packages/metro/src/HmrServer.js
@@ -75,12 +75,24 @@ declare class HmrServer<TClient extends Client> {
75
75
  _handleFileChange(
76
76
  group: ClientGroup,
77
77
  options: {isInitialUpdate: boolean},
78
- changeEvent: null | undefined | {logger: null | undefined | RootPerfLogger},
78
+ changeEvent:
79
+ | null
80
+ | undefined
81
+ | {
82
+ readonly logger: null | undefined | RootPerfLogger;
83
+ readonly changeId?: string;
84
+ },
79
85
  ): Promise<void>;
80
86
  _prepareMessage(
81
87
  group: ClientGroup,
82
88
  options: {isInitialUpdate: boolean},
83
- changeEvent: null | undefined | {logger: null | undefined | RootPerfLogger},
89
+ changeEvent:
90
+ | null
91
+ | undefined
92
+ | {
93
+ readonly logger: null | undefined | RootPerfLogger;
94
+ readonly changeId?: string;
95
+ },
84
96
  ): Promise<HmrUpdateMessage | HmrErrorMessage>;
85
97
  }
86
98
  export default HmrServer;
package/src/HmrServer.js CHANGED
@@ -214,6 +214,10 @@ class HmrServer {
214
214
  case "log-opt-in":
215
215
  client.optedIntoHMR = true;
216
216
  break;
217
+ case "heartbeat":
218
+ debug("Heartbeat received");
219
+ sendFn(String(message));
220
+ break;
217
221
  default:
218
222
  break;
219
223
  }
@@ -265,6 +269,9 @@ class HmrServer {
265
269
  send(sendFns, message);
266
270
  send(sendFns, {
267
271
  type: "update-done",
272
+ body: {
273
+ changeId: changeEvent?.changeId,
274
+ },
268
275
  });
269
276
  log({
270
277
  ...createActionEndEntry(processingHmrChange),