"use strict"; var __assign = (this && this.__assign) || function () { __assign = Object.assign || function(t) { for (var s, i = 1, n = arguments.length; i < n; i++) { s = arguments[i]; for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p]; } return t; }; return __assign.apply(this, arguments); }; var __read = (this && this.__read) || function (o, n) { var m = typeof Symbol === "function" && o[Symbol.iterator]; if (!m) return o; var i = m.call(o), r, ar = [], e; try { while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value); } catch (error) { e = { error: error }; } finally { try { if (r && !r.done && (m = i["return"])) m.call(i); } finally { if (e) throw e.error; } } return ar; }; var __spreadArray = (this && this.__spreadArray) || function (to, from) { for (var i = 0, il = from.length, j = to.length; i < il; i++, j++) to[j] = from[i]; return to; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.TsJestTransformer = exports.CACHE_KEY_EL_SEPARATOR = void 0; var fs_1 = require("fs"); var path_1 = require("path"); var ts_jest_compiler_1 = require("./compiler/ts-jest-compiler"); var config_set_1 = require("./config/config-set"); var constants_1 = require("./constants"); var importer_1 = require("./utils/importer"); var json_1 = require("./utils/json"); var jsonable_value_1 = require("./utils/jsonable-value"); var logger_1 = require("./utils/logger"); var messages_1 = require("./utils/messages"); var sha1_1 = require("./utils/sha1"); var version_checkers_1 = require("./utils/version-checkers"); exports.CACHE_KEY_EL_SEPARATOR = '\x00'; var TsJestTransformer = (function () { function TsJestTransformer() { this._depGraphs = new Map(); this._watchMode = false; this._logger = logger_1.rootLogger.child({ namespace: 'ts-jest-transformer' }); version_checkers_1.VersionCheckers.jest.warn(); this.getCacheKey = this.getCacheKey.bind(this); this.process = this.process.bind(this); this._logger.debug('created new transformer'); process.env.TS_JEST = '1'; } TsJestTransformer.prototype._configsFor = function (transformOptions) { var config = transformOptions.config, cacheFS = transformOptions.cacheFS; var ccs = TsJestTransformer._cachedConfigSets.find(function (cs) { return cs.jestConfig.value === config; }); var configSet; if (ccs) { this._transformCfgStr = ccs.transformerCfgStr; this._compiler = ccs.compiler; this._depGraphs = ccs.depGraphs; this._tsResolvedModulesCachePath = ccs.tsResolvedModulesCachePath; this._watchMode = ccs.watchMode; configSet = ccs.configSet; } else { var serializedJestCfg_1 = json_1.stringify(config); var serializedCcs = TsJestTransformer._cachedConfigSets.find(function (cs) { return cs.jestConfig.serialized === serializedJestCfg_1; }); if (serializedCcs) { serializedCcs.jestConfig.value = config; this._transformCfgStr = serializedCcs.transformerCfgStr; this._compiler = serializedCcs.compiler; this._depGraphs = serializedCcs.depGraphs; this._tsResolvedModulesCachePath = serializedCcs.tsResolvedModulesCachePath; this._watchMode = serializedCcs.watchMode; configSet = serializedCcs.configSet; } else { this._logger.info('no matching config-set found, creating a new one'); configSet = this._createConfigSet(config); var jest_1 = __assign({}, config); jest_1.name = undefined; jest_1.cacheDirectory = undefined; this._transformCfgStr = "" + new jsonable_value_1.JsonableValue(jest_1).serialized + configSet.cacheSuffix; this._createCompiler(configSet, cacheFS); this._getFsCachedResolvedModules(configSet); this._watchMode = process.argv.includes('--watch'); TsJestTransformer._cachedConfigSets.push({ jestConfig: new jsonable_value_1.JsonableValue(config), configSet: configSet, transformerCfgStr: this._transformCfgStr, compiler: this._compiler, depGraphs: this._depGraphs, tsResolvedModulesCachePath: this._tsResolvedModulesCachePath, watchMode: this._watchMode, }); } } return configSet; }; TsJestTransformer.prototype._createConfigSet = function (config) { return new config_set_1.ConfigSet(config); }; TsJestTransformer.prototype._createCompiler = function (configSet, cacheFS) { this._compiler = new ts_jest_compiler_1.TsJestCompiler(configSet, cacheFS); }; TsJestTransformer.prototype.process = function (fileContent, filePath, transformOptions) { this._logger.debug({ fileName: filePath, transformOptions: transformOptions }, 'processing', filePath); var result; var jestConfig = transformOptions.config; var configs = this._configsFor(transformOptions); var hooksFile = process.env.TS_JEST_HOOKS; var hooks; if (hooksFile) { hooksFile = path_1.resolve(configs.cwd, hooksFile); hooks = importer_1.importer.tryTheseOr(hooksFile, {}); } else { hooks = {}; } var shouldStringifyContent = configs.shouldStringifyContent(filePath); var babelJest = shouldStringifyContent ? undefined : configs.babelJestTransformer; var isDefinitionFile = filePath.endsWith(constants_1.DECLARATION_TYPE_EXT); var isJsFile = constants_1.JS_JSX_REGEX.test(filePath); var isTsFile = !isDefinitionFile && constants_1.TS_TSX_REGEX.test(filePath); if (shouldStringifyContent) { result = "module.exports=" + json_1.stringify(fileContent); } else if (isDefinitionFile) { result = ''; } else if (!configs.parsedTsConfig.options.allowJs && isJsFile) { this._logger.warn({ fileName: filePath }, messages_1.interpolate("Got a `.js` file to compile while `allowJs` option is not set to `true` (file: {{path}}). To fix this:\n - if you want TypeScript to process JS files, set `allowJs` to `true` in your TypeScript config (usually tsconfig.json)\n - if you do not want TypeScript to process your `.js` files, in your Jest config change the `transform` key which value is `ts-jest` so that it does not match `.js` files anymore", { path: filePath })); result = fileContent; } else if (isJsFile || isTsFile) { result = this._compiler.getCompiledOutput(fileContent, filePath, { depGraphs: this._depGraphs, supportsStaticESM: transformOptions.supportsStaticESM, watchMode: this._watchMode, }); } else { var message = babelJest ? "Got a unknown file type to compile (file: {{path}}). To fix this, in your Jest config change the `transform` key which value is `ts-jest` so that it does not match this kind of files anymore. If you still want Babel to process it, add another entry to the `transform` option with value `babel-jest` which key matches this type of files." : "Got a unknown file type to compile (file: {{path}}). To fix this, in your Jest config change the `transform` key which value is `ts-jest` so that it does not match this kind of files anymore."; this._logger.warn({ fileName: filePath }, messages_1.interpolate(message, { path: filePath })); result = fileContent; } if (babelJest) { this._logger.debug({ fileName: filePath }, 'calling babel-jest processor'); result = babelJest.process(result, filePath, __assign(__assign({}, transformOptions), { instrument: false })); } if (hooks.afterProcess) { this._logger.debug({ fileName: filePath, hookName: 'afterProcess' }, 'calling afterProcess hook'); var newResult = hooks.afterProcess([fileContent, filePath, jestConfig, transformOptions], result); if (newResult !== undefined) { return newResult; } } return result; }; TsJestTransformer.prototype.getCacheKey = function (fileContent, filePath, transformOptions) { var _a; var configs = this._configsFor(transformOptions); this._logger.debug({ fileName: filePath, transformOptions: transformOptions }, 'computing cache key for', filePath); var _b = transformOptions.instrument, instrument = _b === void 0 ? false : _b; var constructingCacheKeyElements = [ this._transformCfgStr, exports.CACHE_KEY_EL_SEPARATOR, configs.rootDir, exports.CACHE_KEY_EL_SEPARATOR, "instrument:" + (instrument ? 'on' : 'off'), exports.CACHE_KEY_EL_SEPARATOR, fileContent, exports.CACHE_KEY_EL_SEPARATOR, filePath, ]; if (!configs.isolatedModules && this._tsResolvedModulesCachePath) { var resolvedModuleNames = void 0; if (((_a = this._depGraphs.get(filePath)) === null || _a === void 0 ? void 0 : _a.fileContent) === fileContent) { this._logger.debug({ fileName: filePath, transformOptions: transformOptions }, 'getting resolved modules from disk caching or memory caching for', filePath); resolvedModuleNames = this._depGraphs .get(filePath) .resolvedModuleNames.filter(function (moduleName) { return fs_1.existsSync(moduleName); }); } else { this._logger.debug({ fileName: filePath, transformOptions: transformOptions }, 'getting resolved modules from TypeScript API for', filePath); resolvedModuleNames = this._compiler.getResolvedModules(fileContent, filePath, transformOptions.cacheFS); this._depGraphs.set(filePath, { fileContent: fileContent, resolvedModuleNames: resolvedModuleNames, }); fs_1.writeFileSync(this._tsResolvedModulesCachePath, json_1.stringify(__spreadArray([], __read(this._depGraphs)))); } resolvedModuleNames.forEach(function (moduleName) { constructingCacheKeyElements.push(exports.CACHE_KEY_EL_SEPARATOR, moduleName, exports.CACHE_KEY_EL_SEPARATOR, fs_1.statSync(moduleName).mtimeMs.toString()); }); } return sha1_1.sha1.apply(void 0, __spreadArray([], __read(constructingCacheKeyElements))); }; TsJestTransformer.prototype._getFsCachedResolvedModules = function (configSet) { var cacheDir = configSet.tsCacheDir; if (!configSet.isolatedModules && cacheDir) { fs_1.mkdirSync(cacheDir, { recursive: true }); this._tsResolvedModulesCachePath = path_1.join(cacheDir, sha1_1.sha1('ts-jest-resolved-modules', exports.CACHE_KEY_EL_SEPARATOR)); try { var cachedTSResolvedModules = fs_1.readFileSync(this._tsResolvedModulesCachePath, 'utf-8'); this._depGraphs = new Map(json_1.parse(cachedTSResolvedModules)); } catch (e) { } } }; TsJestTransformer._cachedConfigSets = []; return TsJestTransformer; }()); exports.TsJestTransformer = TsJestTransformer;