migrate.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. "use strict";
  2. var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
  3. function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
  4. return new (P || (P = Promise))(function (resolve, reject) {
  5. function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
  6. function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
  7. function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
  8. step((generator = generator.apply(thisArg, _arguments || [])).next());
  9. });
  10. };
  11. var __generator = (this && this.__generator) || function (thisArg, body) {
  12. var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;
  13. return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
  14. function verb(n) { return function (v) { return step([n, v]); }; }
  15. function step(op) {
  16. if (f) throw new TypeError("Generator is already executing.");
  17. while (_) try {
  18. if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
  19. if (y = 0, t) op = [op[0] & 2, t.value];
  20. switch (op[0]) {
  21. case 0: case 1: t = op; break;
  22. case 4: _.label++; return { value: op[1], done: false };
  23. case 5: _.label++; y = op[1]; op = [0]; continue;
  24. case 7: op = _.ops.pop(); _.trys.pop(); continue;
  25. default:
  26. if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
  27. if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
  28. if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
  29. if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
  30. if (t[2]) _.ops.pop();
  31. _.trys.pop(); continue;
  32. }
  33. op = body.call(thisArg, _);
  34. } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
  35. if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
  36. }
  37. };
  38. var __read = (this && this.__read) || function (o, n) {
  39. var m = typeof Symbol === "function" && o[Symbol.iterator];
  40. if (!m) return o;
  41. var i = m.call(o), r, ar = [], e;
  42. try {
  43. while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);
  44. }
  45. catch (error) { e = { error: error }; }
  46. finally {
  47. try {
  48. if (r && !r.done && (m = i["return"])) m.call(i);
  49. }
  50. finally { if (e) throw e.error; }
  51. }
  52. return ar;
  53. };
  54. var __spread = (this && this.__spread) || function () {
  55. for (var ar = [], i = 0; i < arguments.length; i++) ar = ar.concat(__read(arguments[i]));
  56. return ar;
  57. };
  58. Object.defineProperty(exports, "__esModule", { value: true });
  59. var bs_logger_1 = require("bs-logger");
  60. var stringifyJson = require("fast-json-stable-stringify");
  61. var fs_1 = require("fs");
  62. var json5_1 = require("json5");
  63. var path_1 = require("path");
  64. var backports_1 = require("../../util/backports");
  65. var presets_1 = require("../helpers/presets");
  66. exports.run = function (args) { return __awaiter(void 0, void 0, void 0, function () {
  67. var nullLogger, file, filePath, footNotes, name, isPackage, actualConfig, migratedConfig, presetName, preset, jsTransformers, jsWithTs, jsWithBabel, presetValue, migratedValue, presetValue, migratedValue, before, after, stringify, prefix;
  68. return __generator(this, function (_a) {
  69. nullLogger = bs_logger_1.createLogger({ targets: [] });
  70. file = args._[0];
  71. filePath = path_1.resolve(process.cwd(), file);
  72. footNotes = [];
  73. if (!fs_1.existsSync(filePath)) {
  74. throw new Error("Configuration file " + file + " does not exists.");
  75. }
  76. name = path_1.basename(file);
  77. isPackage = name === 'package.json';
  78. if (!/\.(js|json)$/.test(name)) {
  79. throw new TypeError("Configuration file " + file + " must be a JavaScript or JSON file.");
  80. }
  81. actualConfig = require(filePath);
  82. if (isPackage) {
  83. actualConfig = actualConfig.jest;
  84. }
  85. if (!actualConfig)
  86. actualConfig = {};
  87. migratedConfig = backports_1.backportJestConfig(nullLogger, actualConfig);
  88. if (!migratedConfig.preset && args.jestPreset) {
  89. if (args.js) {
  90. presetName = args.js === 'babel' ? presets_1.JestPresetNames.jsWIthBabel : presets_1.JestPresetNames.jsWithTs;
  91. }
  92. else {
  93. jsTransformers = Object.keys(migratedConfig.transform || {}).reduce(function (list, pattern) {
  94. if (RegExp(pattern.replace(/^<rootDir>\/?/, '/dummy-project/')).test('/dummy-project/src/foo.js')) {
  95. var transformer = migratedConfig.transform[pattern];
  96. if (/\bbabel-jest\b/.test(transformer))
  97. transformer = 'babel-jest';
  98. else if (/\ts-jest\b/.test(transformer))
  99. transformer = 'ts-jest';
  100. return __spread(list, [transformer]);
  101. }
  102. return list;
  103. }, []);
  104. jsWithTs = jsTransformers.includes('ts-jest');
  105. jsWithBabel = jsTransformers.includes('babel-jest');
  106. if (jsWithBabel && !jsWithTs) {
  107. presetName = presets_1.JestPresetNames.jsWIthBabel;
  108. }
  109. else if (jsWithTs && !jsWithBabel) {
  110. presetName = presets_1.JestPresetNames.jsWithTs;
  111. }
  112. else {
  113. presetName = presets_1.JestPresetNames.default;
  114. }
  115. }
  116. presetName = presetName || presets_1.JestPresetNames.default;
  117. preset = presets_1.allPresets[presetName];
  118. footNotes.push("Detected preset '" + preset.label + "' as the best matching preset for your configuration.\nVisit https://kulshekhar.github.io/ts-jest/user/config/#jest-preset for more information about presets.\n");
  119. }
  120. else if (migratedConfig.preset && migratedConfig.preset.startsWith('ts-jest')) {
  121. if (args.jestPreset === false) {
  122. delete migratedConfig.preset;
  123. }
  124. else {
  125. preset = presets_1.allPresets[migratedConfig.preset] || presets_1.defaults;
  126. }
  127. }
  128. if (preset)
  129. migratedConfig.preset = preset.name;
  130. if (migratedConfig.moduleFileExtensions && migratedConfig.moduleFileExtensions.length && preset) {
  131. presetValue = dedupSort(preset.value.moduleFileExtensions || []).join('::');
  132. migratedValue = dedupSort(migratedConfig.moduleFileExtensions).join('::');
  133. if (presetValue === migratedValue) {
  134. delete migratedConfig.moduleFileExtensions;
  135. }
  136. }
  137. if (migratedConfig.testRegex && preset) {
  138. migratedConfig.testMatch = null;
  139. }
  140. else if (migratedConfig.testMatch && migratedConfig.testMatch.length && preset) {
  141. presetValue = dedupSort(preset.value.testMatch || []).join('::');
  142. migratedValue = dedupSort(migratedConfig.testMatch).join('::');
  143. if (presetValue === migratedValue) {
  144. delete migratedConfig.testMatch;
  145. }
  146. }
  147. if (migratedConfig.transform) {
  148. Object.keys(migratedConfig.transform).forEach(function (key) {
  149. var val = migratedConfig.transform[key];
  150. if (typeof val === 'string' && /\/?ts-jest(?:\/preprocessor\.js)?$/.test(val)) {
  151. ;
  152. migratedConfig.transform[key] = 'ts-jest';
  153. }
  154. });
  155. }
  156. if (preset &&
  157. migratedConfig.transform &&
  158. stringifyJson(migratedConfig.transform) === stringifyJson(preset.value.transform)) {
  159. delete migratedConfig.transform;
  160. }
  161. cleanupConfig(actualConfig);
  162. cleanupConfig(migratedConfig);
  163. before = stringifyJson(actualConfig);
  164. after = stringifyJson(migratedConfig);
  165. if (after === before) {
  166. process.stderr.write("\nNo migration needed for given Jest configuration\n ");
  167. return [2];
  168. }
  169. stringify = /\.json$/.test(file) ? JSON.stringify : json5_1.stringify;
  170. prefix = /\.json$/.test(file) ? '"jest": ' : 'module.exports = ';
  171. if (preset && migratedConfig.transform) {
  172. footNotes.push("\nI couldn't check if your \"transform\" value is the same as mine which is: " + stringify(preset.value.transform, undefined, ' ') + "\nIf it is the case, you can safely remove the \"transform\" from what I've migrated.\n");
  173. }
  174. process.stderr.write("\nMigrated Jest configuration:\n");
  175. process.stdout.write("" + prefix + stringify(migratedConfig, undefined, ' ') + "\n");
  176. if (footNotes.length) {
  177. process.stderr.write("\n" + footNotes.join('\n') + "\n");
  178. }
  179. return [2];
  180. });
  181. }); };
  182. function cleanupConfig(config) {
  183. if (config.globals) {
  184. if (config.globals['ts-jest'] && Object.keys(config.globals['ts-jest']).length === 0) {
  185. delete config.globals['ts-jest'];
  186. }
  187. if (Object.keys(config.globals).length === 0) {
  188. delete config.globals;
  189. }
  190. }
  191. if (config.transform && Object.keys(config.transform).length === 0) {
  192. delete config.transform;
  193. }
  194. if (config.moduleFileExtensions) {
  195. config.moduleFileExtensions = dedupSort(config.moduleFileExtensions);
  196. if (config.moduleFileExtensions.length === 0)
  197. delete config.moduleFileExtensions;
  198. }
  199. if (config.testMatch) {
  200. config.testMatch = dedupSort(config.testMatch);
  201. if (config.testMatch.length === 0)
  202. delete config.testMatch;
  203. }
  204. if (config.preset === presets_1.JestPresetNames.default)
  205. config.preset = presets_1.defaults.name;
  206. }
  207. function dedupSort(arr) {
  208. return arr
  209. .filter(function (s, i, a) { return a.findIndex(function (e) { return s.toString() === e.toString(); }) === i; })
  210. .sort(function (a, b) { return (a.toString() > b.toString() ? 1 : a.toString() < b.toString() ? -1 : 0); });
  211. }
  212. exports.help = function () { return __awaiter(void 0, void 0, void 0, function () {
  213. return __generator(this, function (_a) {
  214. process.stdout.write("\nUsage:\n ts-jest config:migrate [options] <config-file>\n\nArguments:\n <config-file> Can be a js or json Jest config file. If it is a\n package.json file, the configuration will be read from\n the \"jest\" property.\n\nOptions:\n --js ts|babel Process .js files with ts-jest if 'ts' or with\n babel-jest if 'babel'\n --no-jest-preset Disable the use of Jest presets\n");
  215. return [2];
  216. });
  217. }); };