no-extraneous-dependencies.js 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308
  1. 'use strict';var _slicedToArray = function () {function sliceIterator(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"]) _i["return"]();} finally {if (_d) throw _e;}}return _arr;}return function (arr, i) {if (Array.isArray(arr)) {return arr;} else if (Symbol.iterator in Object(arr)) {return sliceIterator(arr, i);} else {throw new TypeError("Invalid attempt to destructure non-iterable instance");}};}();var _path = require('path');var _path2 = _interopRequireDefault(_path);
  2. var _fs = require('fs');var _fs2 = _interopRequireDefault(_fs);
  3. var _minimatch = require('minimatch');var _minimatch2 = _interopRequireDefault(_minimatch);
  4. var _contextCompat = require('eslint-module-utils/contextCompat');
  5. var _pkgUp = require('eslint-module-utils/pkgUp');var _pkgUp2 = _interopRequireDefault(_pkgUp);
  6. var _resolve = require('eslint-module-utils/resolve');var _resolve2 = _interopRequireDefault(_resolve);
  7. var _moduleVisitor = require('eslint-module-utils/moduleVisitor');var _moduleVisitor2 = _interopRequireDefault(_moduleVisitor);
  8. var _importType = require('../core/importType');var _importType2 = _interopRequireDefault(_importType);
  9. var _packagePath = require('../core/packagePath');
  10. var _docsUrl = require('../docsUrl');var _docsUrl2 = _interopRequireDefault(_docsUrl);function _interopRequireDefault(obj) {return obj && obj.__esModule ? obj : { 'default': obj };}
  11. var depFieldCache = new Map();
  12. function hasKeys() {var obj = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
  13. return Object.keys(obj).length > 0;
  14. }
  15. function arrayOrKeys(arrayOrObject) {
  16. return Array.isArray(arrayOrObject) ? arrayOrObject : Object.keys(arrayOrObject);
  17. }
  18. function readJSON(jsonPath, throwException) {
  19. try {
  20. return JSON.parse(_fs2['default'].readFileSync(jsonPath, 'utf8'));
  21. } catch (err) {
  22. if (throwException) {
  23. throw err;
  24. }
  25. }
  26. }
  27. function extractDepFields(pkg) {
  28. return {
  29. dependencies: pkg.dependencies || {},
  30. devDependencies: pkg.devDependencies || {},
  31. optionalDependencies: pkg.optionalDependencies || {},
  32. peerDependencies: pkg.peerDependencies || {},
  33. // BundledDeps should be in the form of an array, but object notation is also supported by
  34. // `npm`, so we convert it to an array if it is an object
  35. bundledDependencies: arrayOrKeys(pkg.bundleDependencies || pkg.bundledDependencies || []) };
  36. }
  37. function getPackageDepFields(packageJsonPath, throwAtRead) {
  38. if (!depFieldCache.has(packageJsonPath)) {
  39. var packageJson = readJSON(packageJsonPath, throwAtRead);
  40. if (packageJson) {
  41. var depFields = extractDepFields(packageJson);
  42. depFieldCache.set(packageJsonPath, depFields);
  43. }
  44. }
  45. return depFieldCache.get(packageJsonPath);
  46. }
  47. function getDependencies(context, packageDir) {
  48. var paths = [];
  49. try {
  50. var packageContent = {
  51. dependencies: {},
  52. devDependencies: {},
  53. optionalDependencies: {},
  54. peerDependencies: {},
  55. bundledDependencies: [] };
  56. if (packageDir && packageDir.length > 0) {
  57. if (!Array.isArray(packageDir)) {
  58. paths = [_path2['default'].resolve(packageDir)];
  59. } else {
  60. paths = packageDir.map(function (dir) {return _path2['default'].resolve(dir);});
  61. }
  62. }
  63. if (paths.length > 0) {
  64. // use rule config to find package.json
  65. paths.forEach(function (dir) {
  66. var packageJsonPath = _path2['default'].join(dir, 'package.json');
  67. var _packageContent = getPackageDepFields(packageJsonPath, paths.length === 1);
  68. if (_packageContent) {
  69. Object.keys(packageContent).forEach(function (depsKey) {
  70. Object.assign(packageContent[depsKey], _packageContent[depsKey]);
  71. });
  72. }
  73. });
  74. } else {
  75. var packageJsonPath = (0, _pkgUp2['default'])({
  76. cwd: (0, _contextCompat.getPhysicalFilename)(context),
  77. normalize: false });
  78. // use closest package.json
  79. Object.assign(
  80. packageContent,
  81. getPackageDepFields(packageJsonPath, false));
  82. }
  83. if (![
  84. packageContent.dependencies,
  85. packageContent.devDependencies,
  86. packageContent.optionalDependencies,
  87. packageContent.peerDependencies,
  88. packageContent.bundledDependencies].
  89. some(hasKeys)) {
  90. return null;
  91. }
  92. return packageContent;
  93. } catch (e) {
  94. if (paths.length > 0 && e.code === 'ENOENT') {
  95. context.report({
  96. message: 'The package.json file could not be found.',
  97. loc: { line: 0, column: 0 } });
  98. }
  99. if (e.name === 'JSONError' || e instanceof SyntaxError) {
  100. context.report({
  101. message: 'The package.json file could not be parsed: ' + String(e.message),
  102. loc: { line: 0, column: 0 } });
  103. }
  104. return null;
  105. }
  106. }
  107. function missingErrorMessage(packageName) {
  108. return '\'' + String(packageName) + '\' should be listed in the project\'s dependencies. Run \'npm i -S ' + String(packageName) + '\' to add it';
  109. }
  110. function devDepErrorMessage(packageName) {
  111. return '\'' + String(packageName) + '\' should be listed in the project\'s dependencies, not devDependencies.';
  112. }
  113. function optDepErrorMessage(packageName) {
  114. return '\'' + String(packageName) + '\' should be listed in the project\'s dependencies, not optionalDependencies.';
  115. }
  116. function getModuleOriginalName(name) {var _name$split =
  117. name.split('/'),_name$split2 = _slicedToArray(_name$split, 2),first = _name$split2[0],second = _name$split2[1];
  118. return first.startsWith('@') ? String(first) + '/' + String(second) : first;
  119. }
  120. function getModuleRealName(resolved) {
  121. return (0, _packagePath.getFilePackageName)(resolved);
  122. }
  123. function checkDependencyDeclaration(deps, packageName, declarationStatus) {
  124. var newDeclarationStatus = declarationStatus || {
  125. isInDeps: false,
  126. isInDevDeps: false,
  127. isInOptDeps: false,
  128. isInPeerDeps: false,
  129. isInBundledDeps: false };
  130. // in case of sub package.json inside a module
  131. // check the dependencies on all hierarchy
  132. var packageHierarchy = [];
  133. var packageNameParts = packageName ? packageName.split('/') : [];
  134. packageNameParts.forEach(function (namePart, index) {
  135. if (!namePart.startsWith('@')) {
  136. var ancestor = packageNameParts.slice(0, index + 1).join('/');
  137. packageHierarchy.push(ancestor);
  138. }
  139. });
  140. return packageHierarchy.reduce(function (result, ancestorName) {return {
  141. isInDeps: result.isInDeps || deps.dependencies[ancestorName] !== undefined,
  142. isInDevDeps: result.isInDevDeps || deps.devDependencies[ancestorName] !== undefined,
  143. isInOptDeps: result.isInOptDeps || deps.optionalDependencies[ancestorName] !== undefined,
  144. isInPeerDeps: result.isInPeerDeps || deps.peerDependencies[ancestorName] !== undefined,
  145. isInBundledDeps:
  146. result.isInBundledDeps || deps.bundledDependencies.indexOf(ancestorName) !== -1 };},
  147. newDeclarationStatus);
  148. }
  149. function reportIfMissing(context, deps, depsOptions, node, name) {
  150. // Do not report when importing types unless option is enabled
  151. if (
  152. !depsOptions.verifyTypeImports && (
  153. node.importKind === 'type' ||
  154. node.importKind === 'typeof' ||
  155. node.exportKind === 'type' ||
  156. Array.isArray(node.specifiers) && node.specifiers.length && node.specifiers.every(function (specifier) {return specifier.importKind === 'type' || specifier.importKind === 'typeof';})))
  157. {
  158. return;
  159. }
  160. var typeOfImport = (0, _importType2['default'])(name, context);
  161. if (
  162. typeOfImport !== 'external' && (
  163. typeOfImport !== 'internal' || !depsOptions.verifyInternalDeps))
  164. {
  165. return;
  166. }
  167. var resolved = (0, _resolve2['default'])(name, context);
  168. if (!resolved) {return;}
  169. var importPackageName = getModuleOriginalName(name);
  170. var declarationStatus = checkDependencyDeclaration(deps, importPackageName);
  171. if (
  172. declarationStatus.isInDeps ||
  173. depsOptions.allowDevDeps && declarationStatus.isInDevDeps ||
  174. depsOptions.allowPeerDeps && declarationStatus.isInPeerDeps ||
  175. depsOptions.allowOptDeps && declarationStatus.isInOptDeps ||
  176. depsOptions.allowBundledDeps && declarationStatus.isInBundledDeps)
  177. {
  178. return;
  179. }
  180. // test the real name from the resolved package.json
  181. // if not aliased imports (alias/react for example), importPackageName can be misinterpreted
  182. var realPackageName = getModuleRealName(resolved);
  183. if (realPackageName && realPackageName !== importPackageName) {
  184. declarationStatus = checkDependencyDeclaration(deps, realPackageName, declarationStatus);
  185. if (
  186. declarationStatus.isInDeps ||
  187. depsOptions.allowDevDeps && declarationStatus.isInDevDeps ||
  188. depsOptions.allowPeerDeps && declarationStatus.isInPeerDeps ||
  189. depsOptions.allowOptDeps && declarationStatus.isInOptDeps ||
  190. depsOptions.allowBundledDeps && declarationStatus.isInBundledDeps)
  191. {
  192. return;
  193. }
  194. }
  195. if (declarationStatus.isInDevDeps && !depsOptions.allowDevDeps) {
  196. context.report(node, devDepErrorMessage(realPackageName || importPackageName));
  197. return;
  198. }
  199. if (declarationStatus.isInOptDeps && !depsOptions.allowOptDeps) {
  200. context.report(node, optDepErrorMessage(realPackageName || importPackageName));
  201. return;
  202. }
  203. context.report(node, missingErrorMessage(realPackageName || importPackageName));
  204. }
  205. function testConfig(config, filename) {
  206. // Simplest configuration first, either a boolean or nothing.
  207. if (typeof config === 'boolean' || typeof config === 'undefined') {
  208. return config;
  209. }
  210. // Array of globs.
  211. return config.some(function (c) {return (0, _minimatch2['default'])(filename, c) ||
  212. (0, _minimatch2['default'])(filename, _path2['default'].join(process.cwd(), c));});
  213. }
  214. module.exports = {
  215. meta: {
  216. type: 'problem',
  217. docs: {
  218. category: 'Helpful warnings',
  219. description: 'Forbid the use of extraneous packages.',
  220. url: (0, _docsUrl2['default'])('no-extraneous-dependencies') },
  221. schema: [
  222. {
  223. type: 'object',
  224. properties: {
  225. devDependencies: { type: ['boolean', 'array'] },
  226. optionalDependencies: { type: ['boolean', 'array'] },
  227. peerDependencies: { type: ['boolean', 'array'] },
  228. bundledDependencies: { type: ['boolean', 'array'] },
  229. packageDir: { type: ['string', 'array'] },
  230. includeInternal: { type: ['boolean'] },
  231. includeTypes: { type: ['boolean'] } },
  232. additionalProperties: false }] },
  233. create: function () {function create(context) {
  234. var options = context.options[0] || {};
  235. var filename = (0, _contextCompat.getPhysicalFilename)(context);
  236. var deps = getDependencies(context, options.packageDir) || extractDepFields({});
  237. var depsOptions = {
  238. allowDevDeps: testConfig(options.devDependencies, filename) !== false,
  239. allowOptDeps: testConfig(options.optionalDependencies, filename) !== false,
  240. allowPeerDeps: testConfig(options.peerDependencies, filename) !== false,
  241. allowBundledDeps: testConfig(options.bundledDependencies, filename) !== false,
  242. verifyInternalDeps: !!options.includeInternal,
  243. verifyTypeImports: !!options.includeTypes };
  244. return (0, _moduleVisitor2['default'])(function (source, node) {
  245. reportIfMissing(context, deps, depsOptions, node, source.value);
  246. }, { commonjs: true });
  247. }return create;}(),
  248. 'Program:exit': function () {function ProgramExit() {
  249. depFieldCache.clear();
  250. }return ProgramExit;}() };
  251. //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/rules/no-extraneous-dependencies.js"],"names":["depFieldCache","Map","hasKeys","obj","Object","keys","length","arrayOrKeys","arrayOrObject","Array","isArray","readJSON","jsonPath","throwException","JSON","parse","fs","readFileSync","err","extractDepFields","pkg","dependencies","devDependencies","optionalDependencies","peerDependencies","bundledDependencies","bundleDependencies","getPackageDepFields","packageJsonPath","throwAtRead","has","packageJson","depFields","set","get","getDependencies","context","packageDir","paths","packageContent","path","resolve","map","dir","forEach","join","_packageContent","depsKey","assign","cwd","normalize","some","e","code","report","message","loc","line","column","name","SyntaxError","missingErrorMessage","packageName","devDepErrorMessage","optDepErrorMessage","getModuleOriginalName","split","first","second","startsWith","getModuleRealName","resolved","checkDependencyDeclaration","deps","declarationStatus","newDeclarationStatus","isInDeps","isInDevDeps","isInOptDeps","isInPeerDeps","isInBundledDeps","packageHierarchy","packageNameParts","namePart","index","ancestor","slice","push","reduce","result","ancestorName","undefined","indexOf","reportIfMissing","depsOptions","node","verifyTypeImports","importKind","exportKind","specifiers","every","specifier","typeOfImport","verifyInternalDeps","importPackageName","allowDevDeps","allowPeerDeps","allowOptDeps","allowBundledDeps","realPackageName","testConfig","config","filename","c","process","module","exports","meta","type","docs","category","description","url","schema","properties","includeInternal","includeTypes","additionalProperties","create","options","source","value","commonjs","clear"],"mappings":"qoBAAA,4B;AACA,wB;AACA,sC;AACA;AACA,kD;AACA,sD;AACA,kE;;AAEA,gD;AACA;AACA,qC;;AAEA,IAAMA,gBAAgB,IAAIC,GAAJ,EAAtB;;AAEA,SAASC,OAAT,GAA2B,KAAVC,GAAU,uEAAJ,EAAI;AACzB,SAAOC,OAAOC,IAAP,CAAYF,GAAZ,EAAiBG,MAAjB,GAA0B,CAAjC;AACD;;AAED,SAASC,WAAT,CAAqBC,aAArB,EAAoC;AAClC,SAAOC,MAAMC,OAAN,CAAcF,aAAd,IAA+BA,aAA/B,GAA+CJ,OAAOC,IAAP,CAAYG,aAAZ,CAAtD;AACD;;AAED,SAASG,QAAT,CAAkBC,QAAlB,EAA4BC,cAA5B,EAA4C;AAC1C,MAAI;AACF,WAAOC,KAAKC,KAAL,CAAWC,gBAAGC,YAAH,CAAgBL,QAAhB,EAA0B,MAA1B,CAAX,CAAP;AACD,GAFD,CAEE,OAAOM,GAAP,EAAY;AACZ,QAAIL,cAAJ,EAAoB;AAClB,YAAMK,GAAN;AACD;AACF;AACF;;AAED,SAASC,gBAAT,CAA0BC,GAA1B,EAA+B;AAC7B,SAAO;AACLC,kBAAcD,IAAIC,YAAJ,IAAoB,EAD7B;AAELC,qBAAiBF,IAAIE,eAAJ,IAAuB,EAFnC;AAGLC,0BAAsBH,IAAIG,oBAAJ,IAA4B,EAH7C;AAILC,sBAAkBJ,IAAII,gBAAJ,IAAwB,EAJrC;AAKL;AACA;AACAC,yBAAqBlB,YAAYa,IAAIM,kBAAJ,IAA0BN,IAAIK,mBAA9B,IAAqD,EAAjE,CAPhB,EAAP;;AASD;;AAED,SAASE,mBAAT,CAA6BC,eAA7B,EAA8CC,WAA9C,EAA2D;AACzD,MAAI,CAAC7B,cAAc8B,GAAd,CAAkBF,eAAlB,CAAL,EAAyC;AACvC,QAAMG,cAAcpB,SAASiB,eAAT,EAA0BC,WAA1B,CAApB;AACA,QAAIE,WAAJ,EAAiB;AACf,UAAMC,YAAYb,iBAAiBY,WAAjB,CAAlB;AACA/B,oBAAciC,GAAd,CAAkBL,eAAlB,EAAmCI,SAAnC;AACD;AACF;;AAED,SAAOhC,cAAckC,GAAd,CAAkBN,eAAlB,CAAP;AACD;;AAED,SAASO,eAAT,CAAyBC,OAAzB,EAAkCC,UAAlC,EAA8C;AAC5C,MAAIC,QAAQ,EAAZ;AACA,MAAI;AACF,QAAMC,iBAAiB;AACrBlB,oBAAc,EADO;AAErBC,uBAAiB,EAFI;AAGrBC,4BAAsB,EAHD;AAIrBC,wBAAkB,EAJG;AAKrBC,2BAAqB,EALA,EAAvB;;;AAQA,QAAIY,cAAcA,WAAW/B,MAAX,GAAoB,CAAtC,EAAyC;AACvC,UAAI,CAACG,MAAMC,OAAN,CAAc2B,UAAd,CAAL,EAAgC;AAC9BC,gBAAQ,CAACE,kBAAKC,OAAL,CAAaJ,UAAb,CAAD,CAAR;AACD,OAFD,MAEO;AACLC,gBAAQD,WAAWK,GAAX,CAAe,UAACC,GAAD,UAASH,kBAAKC,OAAL,CAAaE,GAAb,CAAT,EAAf,CAAR;AACD;AACF;;AAED,QAAIL,MAAMhC,MAAN,GAAe,CAAnB,EAAsB;AACpB;AACAgC,YAAMM,OAAN,CAAc,UAACD,GAAD,EAAS;AACrB,YAAMf,kBAAkBY,kBAAKK,IAAL,CAAUF,GAAV,EAAe,cAAf,CAAxB;AACA,YAAMG,kBAAkBnB,oBAAoBC,eAApB,EAAqCU,MAAMhC,MAAN,KAAiB,CAAtD,CAAxB;AACA,YAAIwC,eAAJ,EAAqB;AACnB1C,iBAAOC,IAAP,CAAYkC,cAAZ,EAA4BK,OAA5B,CAAoC,UAACG,OAAD,EAAa;AAC/C3C,mBAAO4C,MAAP,CAAcT,eAAeQ,OAAf,CAAd,EAAuCD,gBAAgBC,OAAhB,CAAvC;AACD,WAFD;AAGD;AACF,OARD;AASD,KAXD,MAWO;AACL,UAAMnB,kBAAkB,wBAAM;AAC5BqB,aAAK,wCAAoBb,OAApB,CADuB;AAE5Bc,mBAAW,KAFiB,EAAN,CAAxB;;;AAKA;AACA9C,aAAO4C,MAAP;AACET,oBADF;AAEEZ,0BAAoBC,eAApB,EAAqC,KAArC,CAFF;;AAID;;AAED,QAAI,CAAC;AACHW,mBAAelB,YADZ;AAEHkB,mBAAejB,eAFZ;AAGHiB,mBAAehB,oBAHZ;AAIHgB,mBAAef,gBAJZ;AAKHe,mBAAed,mBALZ;AAMH0B,QANG,CAMEjD,OANF,CAAL,EAMiB;AACf,aAAO,IAAP;AACD;;AAED,WAAOqC,cAAP;AACD,GApDD,CAoDE,OAAOa,CAAP,EAAU;AACV,QAAId,MAAMhC,MAAN,GAAe,CAAf,IAAoB8C,EAAEC,IAAF,KAAW,QAAnC,EAA6C;AAC3CjB,cAAQkB,MAAR,CAAe;AACbC,iBAAS,2CADI;AAEbC,aAAK,EAAEC,MAAM,CAAR,EAAWC,QAAQ,CAAnB,EAFQ,EAAf;;AAID;AACD,QAAIN,EAAEO,IAAF,KAAW,WAAX,IAA0BP,aAAaQ,WAA3C,EAAwD;AACtDxB,cAAQkB,MAAR,CAAe;AACbC,wEAAuDH,EAAEG,OAAzD,CADa;AAEbC,aAAK,EAAEC,MAAM,CAAR,EAAWC,QAAQ,CAAnB,EAFQ,EAAf;;AAID;;AAED,WAAO,IAAP;AACD;AACF;;AAED,SAASG,mBAAT,CAA6BC,WAA7B,EAA0C;AACxC,uBAAWA,WAAX,mFAAyFA,WAAzF;AACD;;AAED,SAASC,kBAAT,CAA4BD,WAA5B,EAAyC;AACvC,uBAAWA,WAAX;AACD;;AAED,SAASE,kBAAT,CAA4BF,WAA5B,EAAyC;AACvC,uBAAWA,WAAX;AACD;;AAED,SAASG,qBAAT,CAA+BN,IAA/B,EAAqC;AACXA,OAAKO,KAAL,CAAW,GAAX,CADW,+CAC5BC,KAD4B,mBACrBC,MADqB;AAEnC,SAAOD,MAAME,UAAN,CAAiB,GAAjB,WAA2BF,KAA3B,iBAAoCC,MAApC,IAA+CD,KAAtD;AACD;;AAED,SAASG,iBAAT,CAA2BC,QAA3B,EAAqC;AACnC,SAAO,qCAAmBA,QAAnB,CAAP;AACD;;AAED,SAASC,0BAAT,CAAoCC,IAApC,EAA0CX,WAA1C,EAAuDY,iBAAvD,EAA0E;AACxE,MAAMC,uBAAuBD,qBAAqB;AAChDE,cAAU,KADsC;AAEhDC,iBAAa,KAFmC;AAGhDC,iBAAa,KAHmC;AAIhDC,kBAAc,KAJkC;AAKhDC,qBAAiB,KAL+B,EAAlD;;;AAQA;AACA;AACA,MAAMC,mBAAmB,EAAzB;AACA,MAAMC,mBAAmBpB,cAAcA,YAAYI,KAAZ,CAAkB,GAAlB,CAAd,GAAuC,EAAhE;AACAgB,mBAAiBtC,OAAjB,CAAyB,UAACuC,QAAD,EAAWC,KAAX,EAAqB;AAC5C,QAAI,CAACD,SAASd,UAAT,CAAoB,GAApB,CAAL,EAA+B;AAC7B,UAAMgB,WAAWH,iBAAiBI,KAAjB,CAAuB,CAAvB,EAA0BF,QAAQ,CAAlC,EAAqCvC,IAArC,CAA0C,GAA1C,CAAjB;AACAoC,uBAAiBM,IAAjB,CAAsBF,QAAtB;AACD;AACF,GALD;;AAOA,SAAOJ,iBAAiBO,MAAjB,CAAwB,UAACC,MAAD,EAASC,YAAT,UAA2B;AACxDd,gBAAUa,OAAOb,QAAP,IAAmBH,KAAKpD,YAAL,CAAkBqE,YAAlB,MAAoCC,SADT;AAExDd,mBAAaY,OAAOZ,WAAP,IAAsBJ,KAAKnD,eAAL,CAAqBoE,YAArB,MAAuCC,SAFlB;AAGxDb,mBAAaW,OAAOX,WAAP,IAAsBL,KAAKlD,oBAAL,CAA0BmE,YAA1B,MAA4CC,SAHvB;AAIxDZ,oBAAcU,OAAOV,YAAP,IAAuBN,KAAKjD,gBAAL,CAAsBkE,YAAtB,MAAwCC,SAJrB;AAKxDX;AACIS,aAAOT,eAAP,IAA0BP,KAAKhD,mBAAL,CAAyBmE,OAAzB,CAAiCF,YAAjC,MAAmD,CAAC,CAN1B,EAA3B,EAAxB;AAOHf,sBAPG,CAAP;AAQD;;AAED,SAASkB,eAAT,CAAyBzD,OAAzB,EAAkCqC,IAAlC,EAAwCqB,WAAxC,EAAqDC,IAArD,EAA2DpC,IAA3D,EAAiE;AAC/D;AACA;AACE,GAACmC,YAAYE,iBAAb;;AAEED,OAAKE,UAAL,KAAoB,MAApB;AACGF,OAAKE,UAAL,KAAoB,QADvB;AAEGF,OAAKG,UAAL,KAAoB,MAFvB;AAGGzF,QAAMC,OAAN,CAAcqF,KAAKI,UAAnB,KAAkCJ,KAAKI,UAAL,CAAgB7F,MAAlD,IAA4DyF,KAAKI,UAAL,CAAgBC,KAAhB,CAAsB,UAACC,SAAD,UAAeA,UAAUJ,UAAV,KAAyB,MAAzB,IAAmCI,UAAUJ,UAAV,KAAyB,QAA3E,EAAtB,CALjE,CADF;;AAQE;AACA;AACD;;AAED,MAAMK,eAAe,6BAAW3C,IAAX,EAAiBvB,OAAjB,CAArB;;AAEA;AACEkE,mBAAiB,UAAjB;AACIA,mBAAiB,UAAjB,IAA+B,CAACR,YAAYS,kBADhD,CADF;AAGE;AACA;AACD;;AAED,MAAMhC,WAAW,0BAAQZ,IAAR,EAAcvB,OAAd,CAAjB;AACA,MAAI,CAACmC,QAAL,EAAe,CAAE,OAAS;;AAE1B,MAAMiC,oBAAoBvC,sBAAsBN,IAAtB,CAA1B;AACA,MAAIe,oBAAoBF,2BAA2BC,IAA3B,EAAiC+B,iBAAjC,CAAxB;;AAEA;AACE9B,oBAAkBE,QAAlB;AACGkB,cAAYW,YAAZ,IAA4B/B,kBAAkBG,WADjD;AAEGiB,cAAYY,aAAZ,IAA6BhC,kBAAkBK,YAFlD;AAGGe,cAAYa,YAAZ,IAA4BjC,kBAAkBI,WAHjD;AAIGgB,cAAYc,gBAAZ,IAAgClC,kBAAkBM,eALvD;AAME;AACA;AACD;;AAED;AACA;AACA,MAAM6B,kBAAkBvC,kBAAkBC,QAAlB,CAAxB;AACA,MAAIsC,mBAAmBA,oBAAoBL,iBAA3C,EAA8D;AAC5D9B,wBAAoBF,2BAA2BC,IAA3B,EAAiCoC,eAAjC,EAAkDnC,iBAAlD,CAApB;;AAEA;AACEA,sBAAkBE,QAAlB;AACGkB,gBAAYW,YAAZ,IAA4B/B,kBAAkBG,WADjD;AAEGiB,gBAAYY,aAAZ,IAA6BhC,kBAAkBK,YAFlD;AAGGe,gBAAYa,YAAZ,IAA4BjC,kBAAkBI,WAHjD;AAIGgB,gBAAYc,gBAAZ,IAAgClC,kBAAkBM,eALvD;AAME;AACA;AACD;AACF;;AAED,MAAIN,kBAAkBG,WAAlB,IAAiC,CAACiB,YAAYW,YAAlD,EAAgE;AAC9DrE,YAAQkB,MAAR,CAAeyC,IAAf,EAAqBhC,mBAAmB8C,mBAAmBL,iBAAtC,CAArB;AACA;AACD;;AAED,MAAI9B,kBAAkBI,WAAlB,IAAiC,CAACgB,YAAYa,YAAlD,EAAgE;AAC9DvE,YAAQkB,MAAR,CAAeyC,IAAf,EAAqB/B,mBAAmB6C,mBAAmBL,iBAAtC,CAArB;AACA;AACD;;AAEDpE,UAAQkB,MAAR,CAAeyC,IAAf,EAAqBlC,oBAAoBgD,mBAAmBL,iBAAvC,CAArB;AACD;;AAED,SAASM,UAAT,CAAoBC,MAApB,EAA4BC,QAA5B,EAAsC;AACpC;AACA,MAAI,OAAOD,MAAP,KAAkB,SAAlB,IAA+B,OAAOA,MAAP,KAAkB,WAArD,EAAkE;AAChE,WAAOA,MAAP;AACD;AACD;AACA,SAAOA,OAAO5D,IAAP,CAAY,UAAC8D,CAAD,UAAO,4BAAUD,QAAV,EAAoBC,CAApB;AACrB,gCAAUD,QAAV,EAAoBxE,kBAAKK,IAAL,CAAUqE,QAAQjE,GAAR,EAAV,EAAyBgE,CAAzB,CAApB,CADc,EAAZ,CAAP;;AAGD;;AAEDE,OAAOC,OAAP,GAAiB;AACfC,QAAM;AACJC,UAAM,SADF;AAEJC,UAAM;AACJC,gBAAU,kBADN;AAEJC,mBAAa,wCAFT;AAGJC,WAAK,0BAAQ,4BAAR,CAHD,EAFF;;;AAQJC,YAAQ;AACN;AACEL,YAAM,QADR;AAEEM,kBAAY;AACVtG,yBAAiB,EAAEgG,MAAM,CAAC,SAAD,EAAY,OAAZ,CAAR,EADP;AAEV/F,8BAAsB,EAAE+F,MAAM,CAAC,SAAD,EAAY,OAAZ,CAAR,EAFZ;AAGV9F,0BAAkB,EAAE8F,MAAM,CAAC,SAAD,EAAY,OAAZ,CAAR,EAHR;AAIV7F,6BAAqB,EAAE6F,MAAM,CAAC,SAAD,EAAY,OAAZ,CAAR,EAJX;AAKVjF,oBAAY,EAAEiF,MAAM,CAAC,QAAD,EAAW,OAAX,CAAR,EALF;AAMVO,yBAAiB,EAAEP,MAAM,CAAC,SAAD,CAAR,EANP;AAOVQ,sBAAc,EAAER,MAAM,CAAC,SAAD,CAAR,EAPJ,EAFd;;AAWES,4BAAsB,KAXxB,EADM,CARJ,EADS;;;;;AA0BfC,QA1Be,+BA0BR5F,OA1BQ,EA0BC;AACd,UAAM6F,UAAU7F,QAAQ6F,OAAR,CAAgB,CAAhB,KAAsB,EAAtC;AACA,UAAMjB,WAAW,wCAAoB5E,OAApB,CAAjB;AACA,UAAMqC,OAAOtC,gBAAgBC,OAAhB,EAAyB6F,QAAQ5F,UAAjC,KAAgDlB,iBAAiB,EAAjB,CAA7D;;AAEA,UAAM2E,cAAc;AAClBW,sBAAcK,WAAWmB,QAAQ3G,eAAnB,EAAoC0F,QAApC,MAAkD,KAD9C;AAElBL,sBAAcG,WAAWmB,QAAQ1G,oBAAnB,EAAyCyF,QAAzC,MAAuD,KAFnD;AAGlBN,uBAAeI,WAAWmB,QAAQzG,gBAAnB,EAAqCwF,QAArC,MAAmD,KAHhD;AAIlBJ,0BAAkBE,WAAWmB,QAAQxG,mBAAnB,EAAwCuF,QAAxC,MAAsD,KAJtD;AAKlBT,4BAAoB,CAAC,CAAC0B,QAAQJ,eALZ;AAMlB7B,2BAAmB,CAAC,CAACiC,QAAQH,YANX,EAApB;;;AASA,aAAO,gCAAc,UAACI,MAAD,EAASnC,IAAT,EAAkB;AACrCF,wBAAgBzD,OAAhB,EAAyBqC,IAAzB,EAA+BqB,WAA/B,EAA4CC,IAA5C,EAAkDmC,OAAOC,KAAzD;AACD,OAFM,EAEJ,EAAEC,UAAU,IAAZ,EAFI,CAAP;AAGD,KA3Cc;;AA6Cf,gBA7Ce,sCA6CE;AACfpI,oBAAcqI,KAAd;AACD,KA/Cc,wBAAjB","file":"no-extraneous-dependencies.js","sourcesContent":["import path from 'path';\nimport fs from 'fs';\nimport minimatch from 'minimatch';\nimport { getPhysicalFilename } from 'eslint-module-utils/contextCompat';\nimport pkgUp from 'eslint-module-utils/pkgUp';\nimport resolve from 'eslint-module-utils/resolve';\nimport moduleVisitor from 'eslint-module-utils/moduleVisitor';\n\nimport importType from '../core/importType';\nimport { getFilePackageName } from '../core/packagePath';\nimport docsUrl from '../docsUrl';\n\nconst depFieldCache = new Map();\n\nfunction hasKeys(obj = {}) {\n  return Object.keys(obj).length > 0;\n}\n\nfunction arrayOrKeys(arrayOrObject) {\n  return Array.isArray(arrayOrObject) ? arrayOrObject : Object.keys(arrayOrObject);\n}\n\nfunction readJSON(jsonPath, throwException) {\n  try {\n    return JSON.parse(fs.readFileSync(jsonPath, 'utf8'));\n  } catch (err) {\n    if (throwException) {\n      throw err;\n    }\n  }\n}\n\nfunction extractDepFields(pkg) {\n  return {\n    dependencies: pkg.dependencies || {},\n    devDependencies: pkg.devDependencies || {},\n    optionalDependencies: pkg.optionalDependencies || {},\n    peerDependencies: pkg.peerDependencies || {},\n    // BundledDeps should be in the form of an array, but object notation is also supported by\n    // `npm`, so we convert it to an array if it is an object\n    bundledDependencies: arrayOrKeys(pkg.bundleDependencies || pkg.bundledDependencies || []),\n  };\n}\n\nfunction getPackageDepFields(packageJsonPath, throwAtRead) {\n  if (!depFieldCache.has(packageJsonPath)) {\n    const packageJson = readJSON(packageJsonPath, throwAtRead);\n    if (packageJson) {\n      const depFields = extractDepFields(packageJson);\n      depFieldCache.set(packageJsonPath, depFields);\n    }\n  }\n\n  return depFieldCache.get(packageJsonPath);\n}\n\nfunction getDependencies(context, packageDir) {\n  let paths = [];\n  try {\n    const packageContent = {\n      dependencies: {},\n      devDependencies: {},\n      optionalDependencies: {},\n      peerDependencies: {},\n      bundledDependencies: [],\n    };\n\n    if (packageDir && packageDir.length > 0) {\n      if (!Array.isArray(packageDir)) {\n        paths = [path.resolve(packageDir)];\n      } else {\n        paths = packageDir.map((dir) => path.resolve(dir));\n      }\n    }\n\n    if (paths.length > 0) {\n      // use rule config to find package.json\n      paths.forEach((dir) => {\n        const packageJsonPath = path.join(dir, 'package.json');\n        const _packageContent = getPackageDepFields(packageJsonPath, paths.length === 1);\n        if (_packageContent) {\n          Object.keys(packageContent).forEach((depsKey) => {\n            Object.assign(packageContent[depsKey], _packageContent[depsKey]);\n          });\n        }\n      });\n    } else {\n      const packageJsonPath = pkgUp({\n        cwd: getPhysicalFilename(context),\n        normalize: false,\n      });\n\n      // use closest package.json\n      Object.assign(\n        packageContent,\n        getPackageDepFields(packageJsonPath, false),\n      );\n    }\n\n    if (![\n      packageContent.dependencies,\n      packageContent.devDependencies,\n      packageContent.optionalDependencies,\n      packageContent.peerDependencies,\n      packageContent.bundledDependencies,\n    ].some(hasKeys)) {\n      return null;\n    }\n\n    return packageContent;\n  } catch (e) {\n    if (paths.length > 0 && e.code === 'ENOENT') {\n      context.report({\n        message: 'The package.json file could not be found.',\n        loc: { line: 0, column: 0 },\n      });\n    }\n    if (e.name === 'JSONError' || e instanceof SyntaxError) {\n      context.report({\n        message: `The package.json file could not be parsed: ${e.message}`,\n        loc: { line: 0, column: 0 },\n      });\n    }\n\n    return null;\n  }\n}\n\nfunction missingErrorMessage(packageName) {\n  return `'${packageName}' should be listed in the project's dependencies. Run 'npm i -S ${packageName}' to add it`;\n}\n\nfunction devDepErrorMessage(packageName) {\n  return `'${packageName}' should be listed in the project's dependencies, not devDependencies.`;\n}\n\nfunction optDepErrorMessage(packageName) {\n  return `'${packageName}' should be listed in the project's dependencies, not optionalDependencies.`;\n}\n\nfunction getModuleOriginalName(name) {\n  const [first, second] = name.split('/');\n  return first.startsWith('@') ? `${first}/${second}` : first;\n}\n\nfunction getModuleRealName(resolved) {\n  return getFilePackageName(resolved);\n}\n\nfunction checkDependencyDeclaration(deps, packageName, declarationStatus) {\n  const newDeclarationStatus = declarationStatus || {\n    isInDeps: false,\n    isInDevDeps: false,\n    isInOptDeps: false,\n    isInPeerDeps: false,\n    isInBundledDeps: false,\n  };\n\n  // in case of sub package.json inside a module\n  // check the dependencies on all hierarchy\n  const packageHierarchy = [];\n  const packageNameParts = packageName ? packageName.split('/') : [];\n  packageNameParts.forEach((namePart, index) => {\n    if (!namePart.startsWith('@')) {\n      const ancestor = packageNameParts.slice(0, index + 1).join('/');\n      packageHierarchy.push(ancestor);\n    }\n  });\n\n  return packageHierarchy.reduce((result, ancestorName) => ({\n    isInDeps: result.isInDeps || deps.dependencies[ancestorName] !== undefined,\n    isInDevDeps: result.isInDevDeps || deps.devDependencies[ancestorName] !== undefined,\n    isInOptDeps: result.isInOptDeps || deps.optionalDependencies[ancestorName] !== undefined,\n    isInPeerDeps: result.isInPeerDeps || deps.peerDependencies[ancestorName] !== undefined,\n    isInBundledDeps:\n        result.isInBundledDeps || deps.bundledDependencies.indexOf(ancestorName) !== -1,\n  }), newDeclarationStatus);\n}\n\nfunction reportIfMissing(context, deps, depsOptions, node, name) {\n  // Do not report when importing types unless option is enabled\n  if (\n    !depsOptions.verifyTypeImports\n    && (\n      node.importKind === 'type'\n      || node.importKind === 'typeof'\n      || node.exportKind === 'type'\n      || Array.isArray(node.specifiers) && node.specifiers.length && node.specifiers.every((specifier) => specifier.importKind === 'type' || specifier.importKind === 'typeof')\n    )\n  ) {\n    return;\n  }\n\n  const typeOfImport = importType(name, context);\n\n  if (\n    typeOfImport !== 'external'\n    && (typeOfImport !== 'internal' || !depsOptions.verifyInternalDeps)\n  ) {\n    return;\n  }\n\n  const resolved = resolve(name, context);\n  if (!resolved) { return; }\n\n  const importPackageName = getModuleOriginalName(name);\n  let declarationStatus = checkDependencyDeclaration(deps, importPackageName);\n\n  if (\n    declarationStatus.isInDeps\n    || depsOptions.allowDevDeps && declarationStatus.isInDevDeps\n    || depsOptions.allowPeerDeps && declarationStatus.isInPeerDeps\n    || depsOptions.allowOptDeps && declarationStatus.isInOptDeps\n    || depsOptions.allowBundledDeps && declarationStatus.isInBundledDeps\n  ) {\n    return;\n  }\n\n  // test the real name from the resolved package.json\n  // if not aliased imports (alias/react for example), importPackageName can be misinterpreted\n  const realPackageName = getModuleRealName(resolved);\n  if (realPackageName && realPackageName !== importPackageName) {\n    declarationStatus = checkDependencyDeclaration(deps, realPackageName, declarationStatus);\n\n    if (\n      declarationStatus.isInDeps\n      || depsOptions.allowDevDeps && declarationStatus.isInDevDeps\n      || depsOptions.allowPeerDeps && declarationStatus.isInPeerDeps\n      || depsOptions.allowOptDeps && declarationStatus.isInOptDeps\n      || depsOptions.allowBundledDeps && declarationStatus.isInBundledDeps\n    ) {\n      return;\n    }\n  }\n\n  if (declarationStatus.isInDevDeps && !depsOptions.allowDevDeps) {\n    context.report(node, devDepErrorMessage(realPackageName || importPackageName));\n    return;\n  }\n\n  if (declarationStatus.isInOptDeps && !depsOptions.allowOptDeps) {\n    context.report(node, optDepErrorMessage(realPackageName || importPackageName));\n    return;\n  }\n\n  context.report(node, missingErrorMessage(realPackageName || importPackageName));\n}\n\nfunction testConfig(config, filename) {\n  // Simplest configuration first, either a boolean or nothing.\n  if (typeof config === 'boolean' || typeof config === 'undefined') {\n    return config;\n  }\n  // Array of globs.\n  return config.some((c) => minimatch(filename, c)\n    || minimatch(filename, path.join(process.cwd(), c)),\n  );\n}\n\nmodule.exports = {\n  meta: {\n    type: 'problem',\n    docs: {\n      category: 'Helpful warnings',\n      description: 'Forbid the use of extraneous packages.',\n      url: docsUrl('no-extraneous-dependencies'),\n    },\n\n    schema: [\n      {\n        type: 'object',\n        properties: {\n          devDependencies: { type: ['boolean', 'array'] },\n          optionalDependencies: { type: ['boolean', 'array'] },\n          peerDependencies: { type: ['boolean', 'array'] },\n          bundledDependencies: { type: ['boolean', 'array'] },\n          packageDir: { type: ['string', 'array'] },\n          includeInternal: { type: ['boolean'] },\n          includeTypes: { type: ['boolean'] },\n        },\n        additionalProperties: false,\n      },\n    ],\n  },\n\n  create(context) {\n    const options = context.options[0] || {};\n    const filename = getPhysicalFilename(context);\n    const deps = getDependencies(context, options.packageDir) || extractDepFields({});\n\n    const depsOptions = {\n      allowDevDeps: testConfig(options.devDependencies, filename) !== false,\n      allowOptDeps: testConfig(options.optionalDependencies, filename) !== false,\n      allowPeerDeps: testConfig(options.peerDependencies, filename) !== false,\n      allowBundledDeps: testConfig(options.bundledDependencies, filename) !== false,\n      verifyInternalDeps: !!options.includeInternal,\n      verifyTypeImports: !!options.includeTypes,\n    };\n\n    return moduleVisitor((source, node) => {\n      reportIfMissing(context, deps, depsOptions, node, source.value);\n    }, { commonjs: true });\n  },\n\n  'Program:exit'() {\n    depFieldCache.clear();\n  },\n};\n"]}