1394 lines
		
	
	
		
			38 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			1394 lines
		
	
	
		
			38 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| 'use strict';
 | |
| 
 | |
| function _child_process() {
 | |
|   const data = require('child_process');
 | |
| 
 | |
|   _child_process = function _child_process() {
 | |
|     return data;
 | |
|   };
 | |
| 
 | |
|   return data;
 | |
| }
 | |
| 
 | |
| function _crypto() {
 | |
|   const data = _interopRequireDefault(require('crypto'));
 | |
| 
 | |
|   _crypto = function _crypto() {
 | |
|     return data;
 | |
|   };
 | |
| 
 | |
|   return data;
 | |
| }
 | |
| 
 | |
| function _events() {
 | |
|   const data = _interopRequireDefault(require('events'));
 | |
| 
 | |
|   _events = function _events() {
 | |
|     return data;
 | |
|   };
 | |
| 
 | |
|   return data;
 | |
| }
 | |
| 
 | |
| function _os() {
 | |
|   const data = _interopRequireDefault(require('os'));
 | |
| 
 | |
|   _os = function _os() {
 | |
|     return data;
 | |
|   };
 | |
| 
 | |
|   return data;
 | |
| }
 | |
| 
 | |
| function _path() {
 | |
|   const data = _interopRequireDefault(require('path'));
 | |
| 
 | |
|   _path = function _path() {
 | |
|     return data;
 | |
|   };
 | |
| 
 | |
|   return data;
 | |
| }
 | |
| 
 | |
| function _sane() {
 | |
|   const data = _interopRequireDefault(require('sane'));
 | |
| 
 | |
|   _sane = function _sane() {
 | |
|     return data;
 | |
|   };
 | |
| 
 | |
|   return data;
 | |
| }
 | |
| 
 | |
| function _invariant() {
 | |
|   const data = _interopRequireDefault(require('invariant'));
 | |
| 
 | |
|   _invariant = function _invariant() {
 | |
|     return data;
 | |
|   };
 | |
| 
 | |
|   return data;
 | |
| }
 | |
| 
 | |
| function _jestSerializer() {
 | |
|   const data = _interopRequireDefault(require('jest-serializer'));
 | |
| 
 | |
|   _jestSerializer = function _jestSerializer() {
 | |
|     return data;
 | |
|   };
 | |
| 
 | |
|   return data;
 | |
| }
 | |
| 
 | |
| function _jestWorker() {
 | |
|   const data = _interopRequireDefault(require('jest-worker'));
 | |
| 
 | |
|   _jestWorker = function _jestWorker() {
 | |
|     return data;
 | |
|   };
 | |
| 
 | |
|   return data;
 | |
| }
 | |
| 
 | |
| var _worker = require('./worker');
 | |
| 
 | |
| var _getMockName = _interopRequireDefault(require('./getMockName'));
 | |
| 
 | |
| var _getPlatformExtension = _interopRequireDefault(
 | |
|   require('./lib/getPlatformExtension')
 | |
| );
 | |
| 
 | |
| var _constants = _interopRequireDefault(require('./constants'));
 | |
| 
 | |
| var _HasteFS = _interopRequireDefault(require('./HasteFS'));
 | |
| 
 | |
| var _ModuleMap = _interopRequireDefault(require('./ModuleMap'));
 | |
| 
 | |
| var _node = _interopRequireDefault(require('./crawlers/node'));
 | |
| 
 | |
| var _normalizePathSep = _interopRequireDefault(
 | |
|   require('./lib/normalizePathSep')
 | |
| );
 | |
| 
 | |
| var _watchman = _interopRequireDefault(require('./crawlers/watchman'));
 | |
| 
 | |
| var _WatchmanWatcher = _interopRequireDefault(require('./lib/WatchmanWatcher'));
 | |
| 
 | |
| var _FSEventsWatcher = _interopRequireDefault(require('./lib/FSEventsWatcher'));
 | |
| 
 | |
| var fastPath = _interopRequireWildcard(require('./lib/fast_path'));
 | |
| 
 | |
| function _interopRequireWildcard(obj) {
 | |
|   if (obj && obj.__esModule) {
 | |
|     return obj;
 | |
|   } else {
 | |
|     var newObj = {};
 | |
|     if (obj != null) {
 | |
|       for (var key in obj) {
 | |
|         if (Object.prototype.hasOwnProperty.call(obj, key)) {
 | |
|           var desc =
 | |
|             Object.defineProperty && Object.getOwnPropertyDescriptor
 | |
|               ? Object.getOwnPropertyDescriptor(obj, key)
 | |
|               : {};
 | |
|           if (desc.get || desc.set) {
 | |
|             Object.defineProperty(newObj, key, desc);
 | |
|           } else {
 | |
|             newObj[key] = obj[key];
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|     newObj.default = obj;
 | |
|     return newObj;
 | |
|   }
 | |
| }
 | |
| 
 | |
| function _interopRequireDefault(obj) {
 | |
|   return obj && obj.__esModule ? obj : {default: obj};
 | |
| }
 | |
| 
 | |
| function _slicedToArray(arr, i) {
 | |
|   return (
 | |
|     _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _nonIterableRest()
 | |
|   );
 | |
| }
 | |
| 
 | |
| function _nonIterableRest() {
 | |
|   throw new TypeError('Invalid attempt to destructure non-iterable instance');
 | |
| }
 | |
| 
 | |
| function _iterableToArrayLimit(arr, i) {
 | |
|   var _arr = [];
 | |
|   var _n = true;
 | |
|   var _d = false;
 | |
|   var _e = undefined;
 | |
|   try {
 | |
|     for (
 | |
|       var _i = arr[Symbol.iterator](), _s;
 | |
|       !(_n = (_s = _i.next()).done);
 | |
|       _n = true
 | |
|     ) {
 | |
|       _arr.push(_s.value);
 | |
|       if (i && _arr.length === i) break;
 | |
|     }
 | |
|   } catch (err) {
 | |
|     _d = true;
 | |
|     _e = err;
 | |
|   } finally {
 | |
|     try {
 | |
|       if (!_n && _i['return'] != null) _i['return']();
 | |
|     } finally {
 | |
|       if (_d) throw _e;
 | |
|     }
 | |
|   }
 | |
|   return _arr;
 | |
| }
 | |
| 
 | |
| function _arrayWithHoles(arr) {
 | |
|   if (Array.isArray(arr)) return arr;
 | |
| }
 | |
| 
 | |
| function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) {
 | |
|   try {
 | |
|     var info = gen[key](arg);
 | |
|     var value = info.value;
 | |
|   } catch (error) {
 | |
|     reject(error);
 | |
|     return;
 | |
|   }
 | |
|   if (info.done) {
 | |
|     resolve(value);
 | |
|   } else {
 | |
|     Promise.resolve(value).then(_next, _throw);
 | |
|   }
 | |
| }
 | |
| 
 | |
| function _asyncToGenerator(fn) {
 | |
|   return function() {
 | |
|     var self = this,
 | |
|       args = arguments;
 | |
|     return new Promise(function(resolve, reject) {
 | |
|       var gen = fn.apply(self, args);
 | |
|       function _next(value) {
 | |
|         asyncGeneratorStep(gen, resolve, reject, _next, _throw, 'next', value);
 | |
|       }
 | |
|       function _throw(err) {
 | |
|         asyncGeneratorStep(gen, resolve, reject, _next, _throw, 'throw', err);
 | |
|       }
 | |
|       _next(undefined);
 | |
|     });
 | |
|   };
 | |
| }
 | |
| 
 | |
| function _defineProperty(obj, key, value) {
 | |
|   if (key in obj) {
 | |
|     Object.defineProperty(obj, key, {
 | |
|       value: value,
 | |
|       enumerable: true,
 | |
|       configurable: true,
 | |
|       writable: true
 | |
|     });
 | |
|   } else {
 | |
|     obj[key] = value;
 | |
|   }
 | |
|   return obj;
 | |
| }
 | |
| 
 | |
| const CHANGE_INTERVAL = 30;
 | |
| const MAX_WAIT_TIME = 240000;
 | |
| 
 | |
| const NODE_MODULES = _path().default.sep + 'node_modules' + _path().default.sep;
 | |
| 
 | |
| const PACKAGE_JSON = _path().default.sep + 'package.json'; // TypeScript doesn't like us importing from outside `rootDir`, but it doesn't
 | |
| // understand `require`.
 | |
| 
 | |
| const _require = require('../package.json'),
 | |
|   VERSION = _require.version;
 | |
| 
 | |
| const canUseWatchman = (() => {
 | |
|   try {
 | |
|     (0, _child_process().execSync)('watchman --version', {
 | |
|       stdio: ['ignore']
 | |
|     });
 | |
|     return true;
 | |
|   } catch (e) {}
 | |
| 
 | |
|   return false;
 | |
| })();
 | |
| 
 | |
| const escapePathSeparator = string =>
 | |
|   _path().default.sep === '\\' ? string.replace(/(\/|\\)/g, '\\\\') : string;
 | |
| 
 | |
| const getWhiteList = list => {
 | |
|   if (list && list.length) {
 | |
|     const newList = list.map(item =>
 | |
|       escapePathSeparator(item.replace(/(\/)/g, _path().default.sep))
 | |
|     );
 | |
|     return new RegExp(
 | |
|       '(' +
 | |
|         escapePathSeparator(NODE_MODULES) +
 | |
|         '(?:' +
 | |
|         newList.join('|') +
 | |
|         ')(?=$|' +
 | |
|         escapePathSeparator(_path().default.sep) +
 | |
|         '))',
 | |
|       'g'
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   return null;
 | |
| };
 | |
| /**
 | |
|  * HasteMap is a JavaScript implementation of Facebook's haste module system.
 | |
|  *
 | |
|  * This implementation is inspired by https://github.com/facebook/node-haste
 | |
|  * and was built with for high-performance in large code repositories with
 | |
|  * hundreds of thousands of files. This implementation is scalable and provides
 | |
|  * predictable performance.
 | |
|  *
 | |
|  * Because the haste map creation and synchronization is critical to startup
 | |
|  * performance and most tasks are blocked by I/O this class makes heavy use of
 | |
|  * synchronous operations. It uses worker processes for parallelizing file
 | |
|  * access and metadata extraction.
 | |
|  *
 | |
|  * The data structures created by `jest-haste-map` can be used directly from the
 | |
|  * cache without further processing. The metadata objects in the `files` and
 | |
|  * `map` objects contain cross-references: a metadata object from one can look
 | |
|  * up the corresponding metadata object in the other map. Note that in most
 | |
|  * projects, the number of files will be greater than the number of haste
 | |
|  * modules one module can refer to many files based on platform extensions.
 | |
|  *
 | |
|  * type HasteMap = {
 | |
|  *   clocks: WatchmanClocks,
 | |
|  *   files: {[filepath: string]: FileMetaData},
 | |
|  *   map: {[id: string]: ModuleMapItem},
 | |
|  *   mocks: {[id: string]: string},
 | |
|  * }
 | |
|  *
 | |
|  * // Watchman clocks are used for query synchronization and file system deltas.
 | |
|  * type WatchmanClocks = {[filepath: string]: string};
 | |
|  *
 | |
|  * type FileMetaData = {
 | |
|  *   id: ?string, // used to look up module metadata objects in `map`.
 | |
|  *   mtime: number, // check for outdated files.
 | |
|  *   size: number, // size of the file in bytes.
 | |
|  *   visited: boolean, // whether the file has been parsed or not.
 | |
|  *   dependencies: Array<string>, // all relative dependencies of this file.
 | |
|  *   sha1: ?string, // SHA-1 of the file, if requested via options.
 | |
|  * };
 | |
|  *
 | |
|  * // Modules can be targeted to a specific platform based on the file name.
 | |
|  * // Example: platform.ios.js and Platform.android.js will both map to the same
 | |
|  * // `Platform` module. The platform should be specified during resolution.
 | |
|  * type ModuleMapItem = {[platform: string]: ModuleMetaData};
 | |
|  *
 | |
|  * //
 | |
|  * type ModuleMetaData = {
 | |
|  *   path: string, // the path to look up the file object in `files`.
 | |
|  *   type: string, // the module type (either `package` or `module`).
 | |
|  * };
 | |
|  *
 | |
|  * Note that the data structures described above are conceptual only. The actual
 | |
|  * implementation uses arrays and constant keys for metadata storage. Instead of
 | |
|  * `{id: 'flatMap', mtime: 3421, size: 42, visited: true, dependencies: []}` the real
 | |
|  * representation is similar to `['flatMap', 3421, 42, 1, []]` to save storage space
 | |
|  * and reduce parse and write time of a big JSON blob.
 | |
|  *
 | |
|  * The HasteMap is created as follows:
 | |
|  *  1. read data from the cache or create an empty structure.
 | |
|  *
 | |
|  *  2. crawl the file system.
 | |
|  *     * empty cache: crawl the entire file system.
 | |
|  *     * cache available:
 | |
|  *       * if watchman is available: get file system delta changes.
 | |
|  *       * if watchman is unavailable: crawl the entire file system.
 | |
|  *     * build metadata objects for every file. This builds the `files` part of
 | |
|  *       the `HasteMap`.
 | |
|  *
 | |
|  *  3. parse and extract metadata from changed files.
 | |
|  *     * this is done in parallel over worker processes to improve performance.
 | |
|  *     * the worst case is to parse all files.
 | |
|  *     * the best case is no file system access and retrieving all data from
 | |
|  *       the cache.
 | |
|  *     * the average case is a small number of changed files.
 | |
|  *
 | |
|  *  4. serialize the new `HasteMap` in a cache file.
 | |
|  *     Worker processes can directly access the cache through `HasteMap.read()`.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| /* eslint-disable-next-line no-redeclare */
 | |
| 
 | |
| class HasteMap extends _events().default {
 | |
|   constructor(options) {
 | |
|     super();
 | |
| 
 | |
|     _defineProperty(this, '_buildPromise', void 0);
 | |
| 
 | |
|     _defineProperty(this, '_cachePath', void 0);
 | |
| 
 | |
|     _defineProperty(this, '_changeInterval', void 0);
 | |
| 
 | |
|     _defineProperty(this, '_console', void 0);
 | |
| 
 | |
|     _defineProperty(this, '_options', void 0);
 | |
| 
 | |
|     _defineProperty(this, '_watchers', void 0);
 | |
| 
 | |
|     _defineProperty(this, '_whitelist', void 0);
 | |
| 
 | |
|     _defineProperty(this, '_worker', void 0);
 | |
| 
 | |
|     this._options = {
 | |
|       cacheDirectory: options.cacheDirectory || _os().default.tmpdir(),
 | |
|       computeDependencies:
 | |
|         options.computeDependencies === undefined
 | |
|           ? true
 | |
|           : options.computeDependencies,
 | |
|       computeSha1: options.computeSha1 || false,
 | |
|       dependencyExtractor: options.dependencyExtractor,
 | |
|       extensions: options.extensions,
 | |
|       forceNodeFilesystemAPI: !!options.forceNodeFilesystemAPI,
 | |
|       hasteImplModulePath: options.hasteImplModulePath,
 | |
|       ignorePattern: options.ignorePattern,
 | |
|       mapper: options.mapper,
 | |
|       maxWorkers: options.maxWorkers,
 | |
|       mocksPattern: options.mocksPattern
 | |
|         ? new RegExp(options.mocksPattern)
 | |
|         : null,
 | |
|       name: options.name,
 | |
|       platforms: options.platforms,
 | |
|       resetCache: options.resetCache,
 | |
|       retainAllFiles: options.retainAllFiles,
 | |
|       rootDir: options.rootDir,
 | |
|       roots: Array.from(new Set(options.roots)),
 | |
|       skipPackageJson: !!options.skipPackageJson,
 | |
|       throwOnModuleCollision: !!options.throwOnModuleCollision,
 | |
|       useWatchman: options.useWatchman == null ? true : options.useWatchman,
 | |
|       watch: !!options.watch
 | |
|     };
 | |
|     this._console = options.console || global.console;
 | |
| 
 | |
|     if (options.ignorePattern && !(options.ignorePattern instanceof RegExp)) {
 | |
|       this._console.warn(
 | |
|         'jest-haste-map: the `ignorePattern` options as a function is being ' +
 | |
|           'deprecated. Provide a RegExp instead. See https://github.com/facebook/jest/pull/4063.'
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     const rootDirHash = _crypto()
 | |
|       .default.createHash('md5')
 | |
|       .update(options.rootDir)
 | |
|       .digest('hex');
 | |
| 
 | |
|     let hasteImplHash = '';
 | |
|     let dependencyExtractorHash = '';
 | |
| 
 | |
|     if (options.hasteImplModulePath) {
 | |
|       const hasteImpl = require(options.hasteImplModulePath);
 | |
| 
 | |
|       if (hasteImpl.getCacheKey) {
 | |
|         hasteImplHash = String(hasteImpl.getCacheKey());
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (options.dependencyExtractor) {
 | |
|       const dependencyExtractor = require(options.dependencyExtractor);
 | |
| 
 | |
|       if (dependencyExtractor.getCacheKey) {
 | |
|         dependencyExtractorHash = String(dependencyExtractor.getCacheKey());
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     this._cachePath = HasteMap.getCacheFilePath(
 | |
|       this._options.cacheDirectory,
 | |
|       `haste-map-${this._options.name}-${rootDirHash}`,
 | |
|       VERSION,
 | |
|       this._options.name,
 | |
|       this._options.roots
 | |
|         .map(root => fastPath.relative(options.rootDir, root))
 | |
|         .join(':'),
 | |
|       this._options.extensions.join(':'),
 | |
|       this._options.platforms.join(':'),
 | |
|       this._options.computeSha1.toString(),
 | |
|       options.mocksPattern || '',
 | |
|       (options.ignorePattern || '').toString(),
 | |
|       hasteImplHash,
 | |
|       dependencyExtractorHash
 | |
|     );
 | |
|     this._whitelist = getWhiteList(options.providesModuleNodeModules);
 | |
|     this._buildPromise = null;
 | |
|     this._watchers = [];
 | |
|     this._worker = null;
 | |
|   }
 | |
| 
 | |
|   static getCacheFilePath(tmpdir, name, ...extra) {
 | |
|     const hash = _crypto()
 | |
|       .default.createHash('md5')
 | |
|       .update(extra.join(''));
 | |
| 
 | |
|     return _path().default.join(
 | |
|       tmpdir,
 | |
|       name.replace(/\W/g, '-') + '-' + hash.digest('hex')
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   getCacheFilePath() {
 | |
|     return this._cachePath;
 | |
|   }
 | |
| 
 | |
|   build() {
 | |
|     var _this = this;
 | |
| 
 | |
|     if (!this._buildPromise) {
 | |
|       this._buildPromise = _asyncToGenerator(function*() {
 | |
|         const data = yield _this._buildFileMap(); // Persist when we don't know if files changed (changedFiles undefined)
 | |
|         // or when we know a file was changed or deleted.
 | |
| 
 | |
|         let hasteMap;
 | |
| 
 | |
|         if (
 | |
|           data.changedFiles === undefined ||
 | |
|           data.changedFiles.size > 0 ||
 | |
|           data.removedFiles.size > 0
 | |
|         ) {
 | |
|           hasteMap = yield _this._buildHasteMap(data);
 | |
| 
 | |
|           _this._persist(hasteMap);
 | |
|         } else {
 | |
|           hasteMap = data.hasteMap;
 | |
|         }
 | |
| 
 | |
|         const rootDir = _this._options.rootDir;
 | |
|         const hasteFS = new _HasteFS.default({
 | |
|           files: hasteMap.files,
 | |
|           rootDir
 | |
|         });
 | |
|         const moduleMap = new _ModuleMap.default({
 | |
|           duplicates: hasteMap.duplicates,
 | |
|           map: hasteMap.map,
 | |
|           mocks: hasteMap.mocks,
 | |
|           rootDir
 | |
|         });
 | |
| 
 | |
|         const __hasteMapForTest =
 | |
|           (process.env.NODE_ENV === 'test' && hasteMap) || null;
 | |
| 
 | |
|         yield _this._watch(hasteMap);
 | |
|         return {
 | |
|           __hasteMapForTest,
 | |
|           hasteFS,
 | |
|           moduleMap
 | |
|         };
 | |
|       })();
 | |
|     }
 | |
| 
 | |
|     return this._buildPromise;
 | |
|   }
 | |
|   /**
 | |
|    * 1. read data from the cache or create an empty structure.
 | |
|    */
 | |
| 
 | |
|   read() {
 | |
|     let hasteMap;
 | |
| 
 | |
|     try {
 | |
|       hasteMap = _jestSerializer().default.readFileSync(this._cachePath);
 | |
|     } catch (err) {
 | |
|       hasteMap = this._createEmptyMap();
 | |
|     }
 | |
| 
 | |
|     return hasteMap;
 | |
|   }
 | |
| 
 | |
|   readModuleMap() {
 | |
|     const data = this.read();
 | |
|     return new _ModuleMap.default({
 | |
|       duplicates: data.duplicates,
 | |
|       map: data.map,
 | |
|       mocks: data.mocks,
 | |
|       rootDir: this._options.rootDir
 | |
|     });
 | |
|   }
 | |
|   /**
 | |
|    * 2. crawl the file system.
 | |
|    */
 | |
| 
 | |
|   _buildFileMap() {
 | |
|     var _this2 = this;
 | |
| 
 | |
|     return _asyncToGenerator(function*() {
 | |
|       let hasteMap;
 | |
| 
 | |
|       try {
 | |
|         const read = _this2._options.resetCache
 | |
|           ? _this2._createEmptyMap
 | |
|           : _this2.read;
 | |
|         hasteMap = yield read.call(_this2);
 | |
|       } catch (_unused) {
 | |
|         hasteMap = _this2._createEmptyMap();
 | |
|       }
 | |
| 
 | |
|       return _this2._crawl(hasteMap);
 | |
|     })();
 | |
|   }
 | |
|   /**
 | |
|    * 3. parse and extract metadata from changed files.
 | |
|    */
 | |
| 
 | |
|   _processFile(hasteMap, map, mocks, filePath, workerOptions) {
 | |
|     const rootDir = this._options.rootDir;
 | |
| 
 | |
|     const setModule = (id, module) => {
 | |
|       let moduleMap = map.get(id);
 | |
| 
 | |
|       if (!moduleMap) {
 | |
|         moduleMap = Object.create(null);
 | |
|         map.set(id, moduleMap);
 | |
|       }
 | |
| 
 | |
|       const platform =
 | |
|         (0, _getPlatformExtension.default)(
 | |
|           module[_constants.default.PATH],
 | |
|           this._options.platforms
 | |
|         ) || _constants.default.GENERIC_PLATFORM;
 | |
| 
 | |
|       const existingModule = moduleMap[platform];
 | |
| 
 | |
|       if (
 | |
|         existingModule &&
 | |
|         existingModule[_constants.default.PATH] !==
 | |
|           module[_constants.default.PATH]
 | |
|       ) {
 | |
|         const method = this._options.throwOnModuleCollision ? 'error' : 'warn';
 | |
| 
 | |
|         this._console[method](
 | |
|           [
 | |
|             'jest-haste-map: Haste module naming collision: ' + id,
 | |
|             '  The following files share their name; please adjust your hasteImpl:',
 | |
|             '    * <rootDir>' +
 | |
|               _path().default.sep +
 | |
|               existingModule[_constants.default.PATH],
 | |
|             '    * <rootDir>' +
 | |
|               _path().default.sep +
 | |
|               module[_constants.default.PATH],
 | |
|             ''
 | |
|           ].join('\n')
 | |
|         );
 | |
| 
 | |
|         if (this._options.throwOnModuleCollision) {
 | |
|           throw new DuplicateError(
 | |
|             existingModule[_constants.default.PATH],
 | |
|             module[_constants.default.PATH]
 | |
|           );
 | |
|         } // We do NOT want consumers to use a module that is ambiguous.
 | |
| 
 | |
|         delete moduleMap[platform];
 | |
| 
 | |
|         if (Object.keys(moduleMap).length === 1) {
 | |
|           map.delete(id);
 | |
|         }
 | |
| 
 | |
|         let dupsByPlatform = hasteMap.duplicates.get(id);
 | |
| 
 | |
|         if (dupsByPlatform == null) {
 | |
|           dupsByPlatform = new Map();
 | |
|           hasteMap.duplicates.set(id, dupsByPlatform);
 | |
|         }
 | |
| 
 | |
|         const dups = new Map([
 | |
|           [module[_constants.default.PATH], module[_constants.default.TYPE]],
 | |
|           [
 | |
|             existingModule[_constants.default.PATH],
 | |
|             existingModule[_constants.default.TYPE]
 | |
|           ]
 | |
|         ]);
 | |
|         dupsByPlatform.set(platform, dups);
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       const dupsByPlatform = hasteMap.duplicates.get(id);
 | |
| 
 | |
|       if (dupsByPlatform != null) {
 | |
|         const dups = dupsByPlatform.get(platform);
 | |
| 
 | |
|         if (dups != null) {
 | |
|           dups.set(
 | |
|             module[_constants.default.PATH],
 | |
|             module[_constants.default.TYPE]
 | |
|           );
 | |
|         }
 | |
| 
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       moduleMap[platform] = module;
 | |
|     };
 | |
| 
 | |
|     const relativeFilePath = fastPath.relative(rootDir, filePath);
 | |
|     const fileMetadata = hasteMap.files.get(relativeFilePath);
 | |
| 
 | |
|     if (!fileMetadata) {
 | |
|       throw new Error(
 | |
|         'jest-haste-map: File to process was not found in the haste map.'
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     const moduleMetadata = hasteMap.map.get(
 | |
|       fileMetadata[_constants.default.ID]
 | |
|     );
 | |
|     const computeSha1 =
 | |
|       this._options.computeSha1 && !fileMetadata[_constants.default.SHA1]; // Callback called when the response from the worker is successful.
 | |
| 
 | |
|     const workerReply = metadata => {
 | |
|       // `1` for truthy values instead of `true` to save cache space.
 | |
|       fileMetadata[_constants.default.VISITED] = 1;
 | |
|       const metadataId = metadata.id;
 | |
|       const metadataModule = metadata.module;
 | |
| 
 | |
|       if (metadataId && metadataModule) {
 | |
|         fileMetadata[_constants.default.ID] = metadataId;
 | |
|         setModule(metadataId, metadataModule);
 | |
|       }
 | |
| 
 | |
|       fileMetadata[_constants.default.DEPENDENCIES] = metadata.dependencies
 | |
|         ? metadata.dependencies.join(_constants.default.DEPENDENCY_DELIM)
 | |
|         : '';
 | |
| 
 | |
|       if (computeSha1) {
 | |
|         fileMetadata[_constants.default.SHA1] = metadata.sha1;
 | |
|       }
 | |
|     }; // Callback called when the response from the worker is an error.
 | |
| 
 | |
|     const workerError = error => {
 | |
|       if (typeof error !== 'object' || !error.message || !error.stack) {
 | |
|         error = new Error(error);
 | |
|         error.stack = ''; // Remove stack for stack-less errors.
 | |
|       } // @ts-ignore: checking error code is OK if error comes from "fs".
 | |
| 
 | |
|       if (!['ENOENT', 'EACCES'].includes(error.code)) {
 | |
|         throw error;
 | |
|       } // If a file cannot be read we remove it from the file list and
 | |
|       // ignore the failure silently.
 | |
| 
 | |
|       hasteMap.files.delete(relativeFilePath);
 | |
|     }; // If we retain all files in the virtual HasteFS representation, we avoid
 | |
|     // reading them if they aren't important (node_modules).
 | |
| 
 | |
|     if (this._options.retainAllFiles && this._isNodeModulesDir(filePath)) {
 | |
|       if (computeSha1) {
 | |
|         return this._getWorker(workerOptions)
 | |
|           .getSha1({
 | |
|             computeDependencies: this._options.computeDependencies,
 | |
|             computeSha1,
 | |
|             dependencyExtractor: this._options.dependencyExtractor,
 | |
|             filePath,
 | |
|             hasteImplModulePath: this._options.hasteImplModulePath,
 | |
|             rootDir
 | |
|           })
 | |
|           .then(workerReply, workerError);
 | |
|       }
 | |
| 
 | |
|       return null;
 | |
|     }
 | |
| 
 | |
|     if (
 | |
|       this._options.mocksPattern &&
 | |
|       this._options.mocksPattern.test(filePath)
 | |
|     ) {
 | |
|       const mockPath = (0, _getMockName.default)(filePath);
 | |
|       const existingMockPath = mocks.get(mockPath);
 | |
| 
 | |
|       if (existingMockPath) {
 | |
|         const secondMockPath = fastPath.relative(rootDir, filePath);
 | |
| 
 | |
|         if (existingMockPath !== secondMockPath) {
 | |
|           const method = this._options.throwOnModuleCollision
 | |
|             ? 'error'
 | |
|             : 'warn';
 | |
| 
 | |
|           this._console[method](
 | |
|             [
 | |
|               'jest-haste-map: duplicate manual mock found: ' + mockPath,
 | |
|               '  The following files share their name; please delete one of them:',
 | |
|               '    * <rootDir>' + _path().default.sep + existingMockPath,
 | |
|               '    * <rootDir>' + _path().default.sep + secondMockPath,
 | |
|               ''
 | |
|             ].join('\n')
 | |
|           );
 | |
| 
 | |
|           if (this._options.throwOnModuleCollision) {
 | |
|             throw new DuplicateError(existingMockPath, secondMockPath);
 | |
|           }
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       mocks.set(mockPath, relativeFilePath);
 | |
|     }
 | |
| 
 | |
|     if (fileMetadata[_constants.default.VISITED]) {
 | |
|       if (!fileMetadata[_constants.default.ID]) {
 | |
|         return null;
 | |
|       }
 | |
| 
 | |
|       if (moduleMetadata != null) {
 | |
|         const platform =
 | |
|           (0, _getPlatformExtension.default)(
 | |
|             filePath,
 | |
|             this._options.platforms
 | |
|           ) || _constants.default.GENERIC_PLATFORM;
 | |
| 
 | |
|         const module = moduleMetadata[platform];
 | |
| 
 | |
|         if (module == null) {
 | |
|           return null;
 | |
|         }
 | |
| 
 | |
|         const moduleId = fileMetadata[_constants.default.ID];
 | |
|         let modulesByPlatform = map.get(moduleId);
 | |
| 
 | |
|         if (!modulesByPlatform) {
 | |
|           modulesByPlatform = Object.create(null);
 | |
|           map.set(moduleId, modulesByPlatform);
 | |
|         }
 | |
| 
 | |
|         modulesByPlatform[platform] = module;
 | |
|         return null;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return this._getWorker(workerOptions)
 | |
|       .worker({
 | |
|         computeDependencies: this._options.computeDependencies,
 | |
|         computeSha1,
 | |
|         dependencyExtractor: this._options.dependencyExtractor,
 | |
|         filePath,
 | |
|         hasteImplModulePath: this._options.hasteImplModulePath,
 | |
|         rootDir
 | |
|       })
 | |
|       .then(workerReply, workerError);
 | |
|   }
 | |
| 
 | |
|   _buildHasteMap(data) {
 | |
|     var _this3 = this;
 | |
| 
 | |
|     return _asyncToGenerator(function*() {
 | |
|       const removedFiles = data.removedFiles,
 | |
|         changedFiles = data.changedFiles,
 | |
|         hasteMap = data.hasteMap; // If any files were removed or we did not track what files changed, process
 | |
|       // every file looking for changes. Otherwise, process only changed files.
 | |
| 
 | |
|       let map;
 | |
|       let mocks;
 | |
|       let filesToProcess;
 | |
| 
 | |
|       if (changedFiles === undefined || removedFiles.size) {
 | |
|         map = new Map();
 | |
|         mocks = new Map();
 | |
|         filesToProcess = hasteMap.files;
 | |
|       } else {
 | |
|         map = hasteMap.map;
 | |
|         mocks = hasteMap.mocks;
 | |
|         filesToProcess = changedFiles;
 | |
|       }
 | |
| 
 | |
|       var _iteratorNormalCompletion = true;
 | |
|       var _didIteratorError = false;
 | |
|       var _iteratorError = undefined;
 | |
| 
 | |
|       try {
 | |
|         for (
 | |
|           var _iterator = removedFiles[Symbol.iterator](), _step;
 | |
|           !(_iteratorNormalCompletion = (_step = _iterator.next()).done);
 | |
|           _iteratorNormalCompletion = true
 | |
|         ) {
 | |
|           const _step$value = _slicedToArray(_step.value, 2),
 | |
|             relativeFilePath = _step$value[0],
 | |
|             fileMetadata = _step$value[1];
 | |
| 
 | |
|           _this3._recoverDuplicates(
 | |
|             hasteMap,
 | |
|             relativeFilePath,
 | |
|             fileMetadata[_constants.default.ID]
 | |
|           );
 | |
|         }
 | |
|       } catch (err) {
 | |
|         _didIteratorError = true;
 | |
|         _iteratorError = err;
 | |
|       } finally {
 | |
|         try {
 | |
|           if (!_iteratorNormalCompletion && _iterator.return != null) {
 | |
|             _iterator.return();
 | |
|           }
 | |
|         } finally {
 | |
|           if (_didIteratorError) {
 | |
|             throw _iteratorError;
 | |
|           }
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       const promises = [];
 | |
|       var _iteratorNormalCompletion2 = true;
 | |
|       var _didIteratorError2 = false;
 | |
|       var _iteratorError2 = undefined;
 | |
| 
 | |
|       try {
 | |
|         for (
 | |
|           var _iterator2 = filesToProcess.keys()[Symbol.iterator](), _step2;
 | |
|           !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done);
 | |
|           _iteratorNormalCompletion2 = true
 | |
|         ) {
 | |
|           const relativeFilePath = _step2.value;
 | |
| 
 | |
|           if (
 | |
|             _this3._options.skipPackageJson &&
 | |
|             relativeFilePath.endsWith(PACKAGE_JSON)
 | |
|           ) {
 | |
|             continue;
 | |
|           } // SHA-1, if requested, should already be present thanks to the crawler.
 | |
| 
 | |
|           const filePath = fastPath.resolve(
 | |
|             _this3._options.rootDir,
 | |
|             relativeFilePath
 | |
|           );
 | |
| 
 | |
|           const promise = _this3._processFile(hasteMap, map, mocks, filePath);
 | |
| 
 | |
|           if (promise) {
 | |
|             promises.push(promise);
 | |
|           }
 | |
|         }
 | |
|       } catch (err) {
 | |
|         _didIteratorError2 = true;
 | |
|         _iteratorError2 = err;
 | |
|       } finally {
 | |
|         try {
 | |
|           if (!_iteratorNormalCompletion2 && _iterator2.return != null) {
 | |
|             _iterator2.return();
 | |
|           }
 | |
|         } finally {
 | |
|           if (_didIteratorError2) {
 | |
|             throw _iteratorError2;
 | |
|           }
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       try {
 | |
|         yield Promise.all(promises);
 | |
| 
 | |
|         _this3._cleanup();
 | |
| 
 | |
|         hasteMap.map = map;
 | |
|         hasteMap.mocks = mocks;
 | |
|         return hasteMap;
 | |
|       } catch (error) {
 | |
|         _this3._cleanup();
 | |
| 
 | |
|         throw error;
 | |
|       }
 | |
|     })();
 | |
|   }
 | |
| 
 | |
|   _cleanup() {
 | |
|     const worker = this._worker; // @ts-ignore
 | |
| 
 | |
|     if (worker && typeof worker.end === 'function') {
 | |
|       // @ts-ignore
 | |
|       worker.end();
 | |
|     }
 | |
| 
 | |
|     this._worker = null;
 | |
|   }
 | |
|   /**
 | |
|    * 4. serialize the new `HasteMap` in a cache file.
 | |
|    */
 | |
| 
 | |
|   _persist(hasteMap) {
 | |
|     _jestSerializer().default.writeFileSync(this._cachePath, hasteMap);
 | |
|   }
 | |
|   /**
 | |
|    * Creates workers or parses files and extracts metadata in-process.
 | |
|    */
 | |
| 
 | |
|   _getWorker(options) {
 | |
|     if (!this._worker) {
 | |
|       if ((options && options.forceInBand) || this._options.maxWorkers <= 1) {
 | |
|         this._worker = {
 | |
|           getSha1: _worker.getSha1,
 | |
|           worker: _worker.worker
 | |
|         };
 | |
|       } else {
 | |
|         // @ts-ignore: assignment of a worker with custom properties.
 | |
|         this._worker = new (_jestWorker()).default(
 | |
|           require.resolve('./worker'),
 | |
|           {
 | |
|             exposedMethods: ['getSha1', 'worker'],
 | |
|             maxRetries: 3,
 | |
|             numWorkers: this._options.maxWorkers
 | |
|           }
 | |
|         );
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return this._worker;
 | |
|   }
 | |
| 
 | |
|   _crawl(hasteMap) {
 | |
|     const options = this._options;
 | |
| 
 | |
|     const ignore = this._ignore.bind(this);
 | |
| 
 | |
|     const crawl =
 | |
|       canUseWatchman && this._options.useWatchman
 | |
|         ? _watchman.default
 | |
|         : _node.default;
 | |
|     const crawlerOptions = {
 | |
|       computeSha1: options.computeSha1,
 | |
|       data: hasteMap,
 | |
|       extensions: options.extensions,
 | |
|       forceNodeFilesystemAPI: options.forceNodeFilesystemAPI,
 | |
|       ignore,
 | |
|       mapper: options.mapper,
 | |
|       rootDir: options.rootDir,
 | |
|       roots: options.roots
 | |
|     };
 | |
| 
 | |
|     const retry = error => {
 | |
|       if (crawl === _watchman.default) {
 | |
|         this._console.warn(
 | |
|           `jest-haste-map: Watchman crawl failed. Retrying once with node ` +
 | |
|             `crawler.\n` +
 | |
|             `  Usually this happens when watchman isn't running. Create an ` +
 | |
|             `empty \`.watchmanconfig\` file in your project's root folder or ` +
 | |
|             `initialize a git or hg repository in your project.\n` +
 | |
|             `  ` +
 | |
|             error
 | |
|         );
 | |
| 
 | |
|         return (0, _node.default)(crawlerOptions).catch(e => {
 | |
|           throw new Error(
 | |
|             `Crawler retry failed:\n` +
 | |
|               `  Original error: ${error.message}\n` +
 | |
|               `  Retry error: ${e.message}\n`
 | |
|           );
 | |
|         });
 | |
|       }
 | |
| 
 | |
|       throw error;
 | |
|     };
 | |
| 
 | |
|     try {
 | |
|       return crawl(crawlerOptions).catch(retry);
 | |
|     } catch (error) {
 | |
|       return retry(error);
 | |
|     }
 | |
|   }
 | |
|   /**
 | |
|    * Watch mode
 | |
|    */
 | |
| 
 | |
|   _watch(hasteMap) {
 | |
|     if (!this._options.watch) {
 | |
|       return Promise.resolve();
 | |
|     } // In watch mode, we'll only warn about module collisions and we'll retain
 | |
|     // all files, even changes to node_modules.
 | |
| 
 | |
|     this._options.throwOnModuleCollision = false;
 | |
|     this._options.retainAllFiles = true; // WatchmanWatcher > FSEventsWatcher > sane.NodeWatcher
 | |
| 
 | |
|     const Watcher =
 | |
|       canUseWatchman && this._options.useWatchman
 | |
|         ? _WatchmanWatcher.default
 | |
|         : _FSEventsWatcher.default.isSupported()
 | |
|         ? _FSEventsWatcher.default
 | |
|         : _sane().default.NodeWatcher;
 | |
|     const extensions = this._options.extensions;
 | |
|     const ignorePattern = this._options.ignorePattern;
 | |
|     const rootDir = this._options.rootDir;
 | |
|     let changeQueue = Promise.resolve();
 | |
|     let eventsQueue = []; // We only need to copy the entire haste map once on every "frame".
 | |
| 
 | |
|     let mustCopy = true;
 | |
| 
 | |
|     const createWatcher = root => {
 | |
|       // @ts-ignore: TODO how? "Cannot use 'new' with an expression whose type lacks a call or construct signature."
 | |
|       const watcher = new Watcher(root, {
 | |
|         dot: false,
 | |
|         glob: extensions.map(extension => '**/*.' + extension),
 | |
|         ignored: ignorePattern
 | |
|       });
 | |
|       return new Promise((resolve, reject) => {
 | |
|         const rejectTimeout = setTimeout(
 | |
|           () => reject(new Error('Failed to start watch mode.')),
 | |
|           MAX_WAIT_TIME
 | |
|         );
 | |
|         watcher.once('ready', () => {
 | |
|           clearTimeout(rejectTimeout);
 | |
|           watcher.on('all', onChange);
 | |
|           resolve(watcher);
 | |
|         });
 | |
|       });
 | |
|     };
 | |
| 
 | |
|     const emitChange = () => {
 | |
|       if (eventsQueue.length) {
 | |
|         mustCopy = true;
 | |
|         const changeEvent = {
 | |
|           eventsQueue,
 | |
|           hasteFS: new _HasteFS.default({
 | |
|             files: hasteMap.files,
 | |
|             rootDir
 | |
|           }),
 | |
|           moduleMap: new _ModuleMap.default({
 | |
|             duplicates: hasteMap.duplicates,
 | |
|             map: hasteMap.map,
 | |
|             mocks: hasteMap.mocks,
 | |
|             rootDir
 | |
|           })
 | |
|         };
 | |
|         this.emit('change', changeEvent);
 | |
|         eventsQueue = [];
 | |
|       }
 | |
|     };
 | |
| 
 | |
|     const onChange = (type, filePath, root, stat) => {
 | |
|       filePath = _path().default.join(
 | |
|         root,
 | |
|         (0, _normalizePathSep.default)(filePath)
 | |
|       );
 | |
| 
 | |
|       if (
 | |
|         (stat && stat.isDirectory()) ||
 | |
|         this._ignore(filePath) ||
 | |
|         !extensions.some(extension => filePath.endsWith(extension))
 | |
|       ) {
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       changeQueue = changeQueue
 | |
|         .then(() => {
 | |
|           // If we get duplicate events for the same file, ignore them.
 | |
|           if (
 | |
|             eventsQueue.find(
 | |
|               event =>
 | |
|                 event.type === type &&
 | |
|                 event.filePath === filePath &&
 | |
|                 ((!event.stat && !stat) ||
 | |
|                   (!!event.stat &&
 | |
|                     !!stat &&
 | |
|                     event.stat.mtime.getTime() === stat.mtime.getTime()))
 | |
|             )
 | |
|           ) {
 | |
|             return null;
 | |
|           }
 | |
| 
 | |
|           if (mustCopy) {
 | |
|             mustCopy = false;
 | |
|             hasteMap = {
 | |
|               clocks: new Map(hasteMap.clocks),
 | |
|               duplicates: new Map(hasteMap.duplicates),
 | |
|               files: new Map(hasteMap.files),
 | |
|               map: new Map(hasteMap.map),
 | |
|               mocks: new Map(hasteMap.mocks)
 | |
|             };
 | |
|           }
 | |
| 
 | |
|           const add = () => {
 | |
|             eventsQueue.push({
 | |
|               filePath,
 | |
|               stat,
 | |
|               type
 | |
|             });
 | |
|             return null;
 | |
|           };
 | |
| 
 | |
|           const relativeFilePath = fastPath.relative(rootDir, filePath);
 | |
|           const fileMetadata = hasteMap.files.get(relativeFilePath); // If it's not an addition, delete the file and all its metadata
 | |
| 
 | |
|           if (fileMetadata != null) {
 | |
|             const moduleName = fileMetadata[_constants.default.ID];
 | |
| 
 | |
|             const platform =
 | |
|               (0, _getPlatformExtension.default)(
 | |
|                 filePath,
 | |
|                 this._options.platforms
 | |
|               ) || _constants.default.GENERIC_PLATFORM;
 | |
| 
 | |
|             hasteMap.files.delete(relativeFilePath);
 | |
|             let moduleMap = hasteMap.map.get(moduleName);
 | |
| 
 | |
|             if (moduleMap != null) {
 | |
|               // We are forced to copy the object because jest-haste-map exposes
 | |
|               // the map as an immutable entity.
 | |
|               moduleMap = copy(moduleMap);
 | |
|               delete moduleMap[platform];
 | |
| 
 | |
|               if (Object.keys(moduleMap).length === 0) {
 | |
|                 hasteMap.map.delete(moduleName);
 | |
|               } else {
 | |
|                 hasteMap.map.set(moduleName, moduleMap);
 | |
|               }
 | |
|             }
 | |
| 
 | |
|             if (
 | |
|               this._options.mocksPattern &&
 | |
|               this._options.mocksPattern.test(filePath)
 | |
|             ) {
 | |
|               const mockName = (0, _getMockName.default)(filePath);
 | |
|               hasteMap.mocks.delete(mockName);
 | |
|             }
 | |
| 
 | |
|             this._recoverDuplicates(hasteMap, relativeFilePath, moduleName);
 | |
|           } // If the file was added or changed,
 | |
|           // parse it and update the haste map.
 | |
| 
 | |
|           if (type === 'add' || type === 'change') {
 | |
|             (0, _invariant().default)(
 | |
|               stat,
 | |
|               'since the file exists or changed, it should have stats'
 | |
|             );
 | |
|             const fileMetadata = [
 | |
|               '',
 | |
|               stat ? stat.mtime.getTime() : -1,
 | |
|               stat ? stat.size : 0,
 | |
|               0,
 | |
|               '',
 | |
|               null
 | |
|             ];
 | |
|             hasteMap.files.set(relativeFilePath, fileMetadata);
 | |
| 
 | |
|             const promise = this._processFile(
 | |
|               hasteMap,
 | |
|               hasteMap.map,
 | |
|               hasteMap.mocks,
 | |
|               filePath,
 | |
|               {
 | |
|                 forceInBand: true
 | |
|               }
 | |
|             ); // Cleanup
 | |
| 
 | |
|             this._cleanup();
 | |
| 
 | |
|             if (promise) {
 | |
|               return promise.then(add);
 | |
|             } else {
 | |
|               // If a file in node_modules has changed,
 | |
|               // emit an event regardless.
 | |
|               add();
 | |
|             }
 | |
|           } else {
 | |
|             add();
 | |
|           }
 | |
| 
 | |
|           return null;
 | |
|         })
 | |
|         .catch(error => {
 | |
|           this._console.error(
 | |
|             `jest-haste-map: watch error:\n  ${error.stack}\n`
 | |
|           );
 | |
|         });
 | |
|     };
 | |
| 
 | |
|     this._changeInterval = setInterval(emitChange, CHANGE_INTERVAL);
 | |
|     return Promise.all(this._options.roots.map(createWatcher)).then(
 | |
|       watchers => {
 | |
|         this._watchers = watchers;
 | |
|       }
 | |
|     );
 | |
|   }
 | |
|   /**
 | |
|    * This function should be called when the file under `filePath` is removed
 | |
|    * or changed. When that happens, we want to figure out if that file was
 | |
|    * part of a group of files that had the same ID. If it was, we want to
 | |
|    * remove it from the group. Furthermore, if there is only one file
 | |
|    * remaining in the group, then we want to restore that single file as the
 | |
|    * correct resolution for its ID, and cleanup the duplicates index.
 | |
|    */
 | |
| 
 | |
|   _recoverDuplicates(hasteMap, relativeFilePath, moduleName) {
 | |
|     let dupsByPlatform = hasteMap.duplicates.get(moduleName);
 | |
| 
 | |
|     if (dupsByPlatform == null) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     const platform =
 | |
|       (0, _getPlatformExtension.default)(
 | |
|         relativeFilePath,
 | |
|         this._options.platforms
 | |
|       ) || _constants.default.GENERIC_PLATFORM;
 | |
| 
 | |
|     let dups = dupsByPlatform.get(platform);
 | |
| 
 | |
|     if (dups == null) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     dupsByPlatform = copyMap(dupsByPlatform);
 | |
|     hasteMap.duplicates.set(moduleName, dupsByPlatform);
 | |
|     dups = copyMap(dups);
 | |
|     dupsByPlatform.set(platform, dups);
 | |
|     dups.delete(relativeFilePath);
 | |
| 
 | |
|     if (dups.size !== 1) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     const uniqueModule = dups.entries().next().value;
 | |
| 
 | |
|     if (!uniqueModule) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     let dedupMap = hasteMap.map.get(moduleName);
 | |
| 
 | |
|     if (dedupMap == null) {
 | |
|       dedupMap = Object.create(null);
 | |
|       hasteMap.map.set(moduleName, dedupMap);
 | |
|     }
 | |
| 
 | |
|     dedupMap[platform] = uniqueModule;
 | |
|     dupsByPlatform.delete(platform);
 | |
| 
 | |
|     if (dupsByPlatform.size === 0) {
 | |
|       hasteMap.duplicates.delete(moduleName);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   end() {
 | |
|     // @ts-ignore: TODO TS cannot decide if `setInterval` and `clearInterval` comes from NodeJS or the DOM
 | |
|     clearInterval(this._changeInterval);
 | |
| 
 | |
|     if (!this._watchers.length) {
 | |
|       return Promise.resolve();
 | |
|     }
 | |
| 
 | |
|     return Promise.all(
 | |
|       this._watchers.map(
 | |
|         watcher => new Promise(resolve => watcher.close(resolve))
 | |
|       )
 | |
|     ).then(() => {
 | |
|       this._watchers = [];
 | |
|     });
 | |
|   }
 | |
|   /**
 | |
|    * Helpers
 | |
|    */
 | |
| 
 | |
|   _ignore(filePath) {
 | |
|     const ignorePattern = this._options.ignorePattern;
 | |
|     const ignoreMatched =
 | |
|       ignorePattern instanceof RegExp
 | |
|         ? ignorePattern.test(filePath)
 | |
|         : ignorePattern && ignorePattern(filePath);
 | |
|     return (
 | |
|       ignoreMatched ||
 | |
|       (!this._options.retainAllFiles && this._isNodeModulesDir(filePath))
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   _isNodeModulesDir(filePath) {
 | |
|     if (!filePath.includes(NODE_MODULES)) {
 | |
|       return false;
 | |
|     }
 | |
| 
 | |
|     if (this._whitelist) {
 | |
|       const whitelist = this._whitelist;
 | |
|       const match = whitelist.exec(filePath);
 | |
|       const matchEndIndex = whitelist.lastIndex;
 | |
|       whitelist.lastIndex = 0;
 | |
| 
 | |
|       if (!match) {
 | |
|         return true;
 | |
|       }
 | |
| 
 | |
|       const filePathInPackage = filePath.substr(matchEndIndex);
 | |
|       return filePathInPackage.startsWith(NODE_MODULES);
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   _createEmptyMap() {
 | |
|     return {
 | |
|       clocks: new Map(),
 | |
|       duplicates: new Map(),
 | |
|       files: new Map(),
 | |
|       map: new Map(),
 | |
|       mocks: new Map()
 | |
|     };
 | |
|   }
 | |
| }
 | |
| 
 | |
| _defineProperty(HasteMap, 'H', void 0);
 | |
| 
 | |
| _defineProperty(HasteMap, 'DuplicateError', void 0);
 | |
| 
 | |
| _defineProperty(HasteMap, 'ModuleMap', void 0);
 | |
| 
 | |
| class DuplicateError extends Error {
 | |
|   constructor(mockPath1, mockPath2) {
 | |
|     super('Duplicated files or mocks. Please check the console for more info');
 | |
| 
 | |
|     _defineProperty(this, 'mockPath1', void 0);
 | |
| 
 | |
|     _defineProperty(this, 'mockPath2', void 0);
 | |
| 
 | |
|     this.mockPath1 = mockPath1;
 | |
|     this.mockPath2 = mockPath2;
 | |
|   }
 | |
| }
 | |
| 
 | |
| function copy(object) {
 | |
|   return Object.assign(Object.create(null), object);
 | |
| }
 | |
| 
 | |
| function copyMap(input) {
 | |
|   return new Map(input);
 | |
| }
 | |
| 
 | |
| HasteMap.H = _constants.default;
 | |
| HasteMap.DuplicateError = DuplicateError;
 | |
| HasteMap.ModuleMap = _ModuleMap.default;
 | |
| module.exports = HasteMap;
 |