UNPKG

ts-jest

Version:

A Jest transformer with source map support that lets you use Jest to test projects written in TypeScript

147 lines (146 loc) 5.82 kB
"use strict"; var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) { var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d; if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc); else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r; return c > 3 && r && Object.defineProperty(target, key, r), r; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.importer = exports.Importer = void 0; exports.__requireModule = __requireModule; const logger_1 = require("./logger"); const memoize_1 = require("./memoize"); const messages_1 = require("./messages"); const logger = logger_1.rootLogger.child({ namespace: 'Importer' }); /** * @internal */ class Importer { static get instance() { logger.debug('creating Importer singleton'); return new Importer(); } babelJest(why) { return this._import(why, 'babel-jest'); } babelCore(why) { return this._import(why, '@babel/core'); } typescript(why, which) { return this._import(why, which); } esBuild(why) { return this._import(why, 'esbuild'); } tryThese(moduleName, ...fallbacks) { let name; let loaded; const tries = [moduleName, ...fallbacks]; while ((name = tries.shift()) !== undefined) { const req = requireWrapper(name); // remove exports from what we're going to log // eslint-disable-next-line @typescript-eslint/no-explicit-any const contextReq = { ...req }; delete contextReq.exports; if (req.exists) { // module exists loaded = req; if (req.error) { logger.error({ requireResult: contextReq }, `failed loading module '${name}'`, req.error.message); } else { logger.debug({ requireResult: contextReq }, 'loaded module', name); } break; } else { // module does not exists in the path logger.debug({ requireResult: contextReq }, `module '${name}' not found`); } } // return the loaded one, could be one that has been loaded, or one which has failed during load // but not one which does not exists return loaded; } tryTheseOr(moduleNames, missingResult, allowLoadError = false) { const args = Array.isArray(moduleNames) ? moduleNames : [moduleNames]; const result = this.tryThese(...args); if (!result) return missingResult; if (!result.error) return result.exports; if (allowLoadError) return missingResult; throw result.error; } _import(why, moduleName, { alternatives = [], installTip = moduleName } = {}) { // try to load any of the alternative after trying main one const res = this.tryThese(moduleName, ...alternatives); // if we could load one, return it if (res?.exists) { if (!res.error) return res.exports; // it could not load because of a failure while importing, but it exists throw new Error((0, messages_1.interpolate)("Loading module {{module}} failed with error: {{error}}" /* Errors.LoadingModuleFailed */, { module: res.given, error: res.error.message })); } // if it couldn't load, build a nice error message so the user can fix it by himself const msg = alternatives.length ? "Unable to load any of these modules: {{module}}. {{reason}}. To fix it:\n{{fix}}" /* Errors.UnableToLoadAnyModule */ : "Unable to load the module {{module}}. {{reason}} To fix it:\n{{fix}}" /* Errors.UnableToLoadOneModule */; const loadModule = [moduleName, ...alternatives].map((m) => `"${m}"`).join(', '); if (typeof installTip === 'string') { installTip = [{ module: installTip, label: `install "${installTip}"` }]; } const fix = installTip .map((tip) => ` ${installTip.length === 1 ? '↳' : '•'} ${(0, messages_1.interpolate)("{{label}}: `npm i -D {{module}}` (or `yarn add --dev {{module}}`)" /* Helps.FixMissingModule */, tip)}`) .join('\n'); throw new Error((0, messages_1.interpolate)(msg, { module: loadModule, reason: why, fix, })); } } exports.Importer = Importer; __decorate([ (0, memoize_1.Memoize)((...args) => args.join(':')) ], Importer.prototype, "tryThese", null); __decorate([ (0, memoize_1.Memoize)() ], Importer, "instance", null); /** * @internal */ exports.importer = Importer.instance; function requireWrapper(moduleName) { let path; let exists = false; try { path = resolveModule(moduleName); exists = true; } catch (error) { return { error: error, exists, given: moduleName }; } const result = { exists, path, given: moduleName }; try { result.exports = requireModule(path); } catch { try { result.exports = requireModule(moduleName); } catch (error) { result.error = error; } } return result; } let requireModule = (mod) => require(mod); let resolveModule = (mod) => require.resolve(mod, { paths: [process.cwd(), __dirname] }); /** * @internal */ // so that we can test easier function __requireModule(localRequire, localResolve) { requireModule = localRequire; resolveModule = localResolve; }