no-extraneous-dependencies.js 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. 'use strict';
  2. var _path = require('path');
  3. var _path2 = _interopRequireDefault(_path);
  4. var _fs = require('fs');
  5. var _fs2 = _interopRequireDefault(_fs);
  6. var _lodash = require('lodash');
  7. var _readPkgUp = require('read-pkg-up');
  8. var _readPkgUp2 = _interopRequireDefault(_readPkgUp);
  9. var _minimatch = require('minimatch');
  10. var _minimatch2 = _interopRequireDefault(_minimatch);
  11. var _resolve = require('eslint-module-utils/resolve');
  12. var _resolve2 = _interopRequireDefault(_resolve);
  13. var _importType = require('../core/importType');
  14. var _importType2 = _interopRequireDefault(_importType);
  15. var _staticRequire = require('../core/staticRequire');
  16. var _staticRequire2 = _interopRequireDefault(_staticRequire);
  17. var _docsUrl = require('../docsUrl');
  18. var _docsUrl2 = _interopRequireDefault(_docsUrl);
  19. function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
  20. function hasKeys() {
  21. let obj = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
  22. return Object.keys(obj).length > 0;
  23. }
  24. function extractDepFields(pkg) {
  25. return {
  26. dependencies: pkg.dependencies || {},
  27. devDependencies: pkg.devDependencies || {},
  28. optionalDependencies: pkg.optionalDependencies || {},
  29. peerDependencies: pkg.peerDependencies || {}
  30. };
  31. }
  32. function getDependencies(context, packageDir) {
  33. let paths = [];
  34. try {
  35. const packageContent = {
  36. dependencies: {},
  37. devDependencies: {},
  38. optionalDependencies: {},
  39. peerDependencies: {}
  40. };
  41. if (!(0, _lodash.isEmpty)(packageDir)) {
  42. if (!(0, _lodash.isArray)(packageDir)) {
  43. paths = [_path2.default.resolve(packageDir)];
  44. } else {
  45. paths = packageDir.map(dir => _path2.default.resolve(dir));
  46. }
  47. }
  48. if (!(0, _lodash.isEmpty)(paths)) {
  49. // use rule config to find package.json
  50. paths.forEach(dir => {
  51. Object.assign(packageContent, extractDepFields(JSON.parse(_fs2.default.readFileSync(_path2.default.join(dir, 'package.json'), 'utf8'))));
  52. });
  53. } else {
  54. // use closest package.json
  55. Object.assign(packageContent, extractDepFields(_readPkgUp2.default.sync({ cwd: context.getFilename(), normalize: false }).pkg));
  56. }
  57. if (![packageContent.dependencies, packageContent.devDependencies, packageContent.optionalDependencies, packageContent.peerDependencies].some(hasKeys)) {
  58. return null;
  59. }
  60. return packageContent;
  61. } catch (e) {
  62. if (!(0, _lodash.isEmpty)(paths) && e.code === 'ENOENT') {
  63. context.report({
  64. message: 'The package.json file could not be found.',
  65. loc: { line: 0, column: 0 }
  66. });
  67. }
  68. if (e.name === 'JSONError' || e instanceof SyntaxError) {
  69. context.report({
  70. message: 'The package.json file could not be parsed: ' + e.message,
  71. loc: { line: 0, column: 0 }
  72. });
  73. }
  74. return null;
  75. }
  76. }
  77. function missingErrorMessage(packageName) {
  78. return `'${packageName}' should be listed in the project's dependencies. ` + `Run 'npm i -S ${packageName}' to add it`;
  79. }
  80. function devDepErrorMessage(packageName) {
  81. return `'${packageName}' should be listed in the project's dependencies, not devDependencies.`;
  82. }
  83. function optDepErrorMessage(packageName) {
  84. return `'${packageName}' should be listed in the project's dependencies, ` + `not optionalDependencies.`;
  85. }
  86. function reportIfMissing(context, deps, depsOptions, node, name) {
  87. // Do not report when importing types
  88. if (node.importKind === 'type') {
  89. return;
  90. }
  91. if ((0, _importType2.default)(name, context) !== 'external') {
  92. return;
  93. }
  94. const resolved = (0, _resolve2.default)(name, context);
  95. if (!resolved) {
  96. return;
  97. }
  98. const splitName = name.split('/');
  99. const packageName = splitName[0][0] === '@' ? splitName.slice(0, 2).join('/') : splitName[0];
  100. const isInDeps = deps.dependencies[packageName] !== undefined;
  101. const isInDevDeps = deps.devDependencies[packageName] !== undefined;
  102. const isInOptDeps = deps.optionalDependencies[packageName] !== undefined;
  103. const isInPeerDeps = deps.peerDependencies[packageName] !== undefined;
  104. if (isInDeps || depsOptions.allowDevDeps && isInDevDeps || depsOptions.allowPeerDeps && isInPeerDeps || depsOptions.allowOptDeps && isInOptDeps) {
  105. return;
  106. }
  107. if (isInDevDeps && !depsOptions.allowDevDeps) {
  108. context.report(node, devDepErrorMessage(packageName));
  109. return;
  110. }
  111. if (isInOptDeps && !depsOptions.allowOptDeps) {
  112. context.report(node, optDepErrorMessage(packageName));
  113. return;
  114. }
  115. context.report(node, missingErrorMessage(packageName));
  116. }
  117. function testConfig(config, filename) {
  118. // Simplest configuration first, either a boolean or nothing.
  119. if (typeof config === 'boolean' || typeof config === 'undefined') {
  120. return config;
  121. }
  122. // Array of globs.
  123. return config.some(c => (0, _minimatch2.default)(filename, c) || (0, _minimatch2.default)(filename, _path2.default.join(process.cwd(), c)));
  124. }
  125. module.exports = {
  126. meta: {
  127. docs: {
  128. url: (0, _docsUrl2.default)('no-extraneous-dependencies')
  129. },
  130. schema: [{
  131. 'type': 'object',
  132. 'properties': {
  133. 'devDependencies': { 'type': ['boolean', 'array'] },
  134. 'optionalDependencies': { 'type': ['boolean', 'array'] },
  135. 'peerDependencies': { 'type': ['boolean', 'array'] },
  136. 'packageDir': { 'type': ['string', 'array'] }
  137. },
  138. 'additionalProperties': false
  139. }]
  140. },
  141. create: function (context) {
  142. const options = context.options[0] || {};
  143. const filename = context.getFilename();
  144. const deps = getDependencies(context, options.packageDir);
  145. if (!deps) {
  146. return {};
  147. }
  148. const depsOptions = {
  149. allowDevDeps: testConfig(options.devDependencies, filename) !== false,
  150. allowOptDeps: testConfig(options.optionalDependencies, filename) !== false,
  151. allowPeerDeps: testConfig(options.peerDependencies, filename) !== false
  152. // todo: use module visitor from module-utils core
  153. };return {
  154. ImportDeclaration: function (node) {
  155. reportIfMissing(context, deps, depsOptions, node, node.source.value);
  156. },
  157. CallExpression: function handleRequires(node) {
  158. if ((0, _staticRequire2.default)(node)) {
  159. reportIfMissing(context, deps, depsOptions, node, node.arguments[0].value);
  160. }
  161. }
  162. };
  163. }
  164. };
  165. //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["rules/no-extraneous-dependencies.js"],"names":["hasKeys","obj","Object","keys","length","extractDepFields","pkg","dependencies","devDependencies","optionalDependencies","peerDependencies","getDependencies","context","packageDir","paths","packageContent","path","resolve","map","dir","forEach","assign","JSON","parse","fs","readFileSync","join","readPkgUp","sync","cwd","getFilename","normalize","some","e","code","report","message","loc","line","column","name","SyntaxError","missingErrorMessage","packageName","devDepErrorMessage","optDepErrorMessage","reportIfMissing","deps","depsOptions","node","importKind","resolved","splitName","split","slice","isInDeps","undefined","isInDevDeps","isInOptDeps","isInPeerDeps","allowDevDeps","allowPeerDeps","allowOptDeps","testConfig","config","filename","c","process","module","exports","meta","docs","url","schema","create","options","ImportDeclaration","source","value","CallExpression","handleRequires","arguments"],"mappings":"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","file":"rules/no-extraneous-dependencies.js","sourcesContent":["import path from 'path'\nimport fs from 'fs'\nimport { isArray, isEmpty } from 'lodash'\nimport readPkgUp from 'read-pkg-up'\nimport minimatch from 'minimatch'\nimport resolve from 'eslint-module-utils/resolve'\nimport importType from '../core/importType'\nimport isStaticRequire from '../core/staticRequire'\nimport docsUrl from '../docsUrl'\n\nfunction hasKeys(obj = {}) {\n  return Object.keys(obj).length > 0\n}\n\nfunction extractDepFields(pkg) {\n  return {\n    dependencies: pkg.dependencies || {},\n    devDependencies: pkg.devDependencies || {},\n    optionalDependencies: pkg.optionalDependencies || {},\n    peerDependencies: pkg.peerDependencies || {},\n  }\n}\n\nfunction getDependencies(context, packageDir) {\n  let paths = []\n  try {\n    const packageContent = {\n      dependencies: {},\n      devDependencies: {},\n      optionalDependencies: {},\n      peerDependencies: {},\n    }\n\n    if (!isEmpty(packageDir)) {\n      if (!isArray(packageDir)) {\n        paths = [path.resolve(packageDir)]\n      } else {\n        paths = packageDir.map(dir => path.resolve(dir))\n      }\n    }\n\n    if (!isEmpty(paths)) {\n      // use rule config to find package.json\n      paths.forEach(dir => {\n        Object.assign(packageContent, extractDepFields(\n          JSON.parse(fs.readFileSync(path.join(dir, 'package.json'), 'utf8'))\n        ))\n      })\n    } else {\n      // use closest package.json\n      Object.assign(\n        packageContent,\n        extractDepFields(\n          readPkgUp.sync({cwd: context.getFilename(), normalize: false}).pkg\n        )\n      )\n    }\n\n    if (![\n      packageContent.dependencies,\n      packageContent.devDependencies,\n      packageContent.optionalDependencies,\n      packageContent.peerDependencies,\n    ].some(hasKeys)) {\n      return null\n    }\n\n    return packageContent\n  } catch (e) {\n    if (!isEmpty(paths) && 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. ` +\n    `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, ` +\n    `not optionalDependencies.`\n}\n\nfunction reportIfMissing(context, deps, depsOptions, node, name) {\n  // Do not report when importing types\n  if (node.importKind === 'type') {\n    return\n  }\n\n  if (importType(name, context) !== 'external') {\n    return\n  }\n\n  const resolved = resolve(name, context)\n  if (!resolved) { return }\n\n  const splitName = name.split('/')\n  const packageName = splitName[0][0] === '@'\n    ? splitName.slice(0, 2).join('/')\n    : splitName[0]\n  const isInDeps = deps.dependencies[packageName] !== undefined\n  const isInDevDeps = deps.devDependencies[packageName] !== undefined\n  const isInOptDeps = deps.optionalDependencies[packageName] !== undefined\n  const isInPeerDeps = deps.peerDependencies[packageName] !== undefined\n\n  if (isInDeps ||\n    (depsOptions.allowDevDeps && isInDevDeps) ||\n    (depsOptions.allowPeerDeps && isInPeerDeps) ||\n    (depsOptions.allowOptDeps && isInOptDeps)\n  ) {\n    return\n  }\n\n  if (isInDevDeps && !depsOptions.allowDevDeps) {\n    context.report(node, devDepErrorMessage(packageName))\n    return\n  }\n\n  if (isInOptDeps && !depsOptions.allowOptDeps) {\n    context.report(node, optDepErrorMessage(packageName))\n    return\n  }\n\n  context.report(node, missingErrorMessage(packageName))\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 => (\n    minimatch(filename, c) ||\n    minimatch(filename, path.join(process.cwd(), c))\n  ))\n}\n\nmodule.exports = {\n  meta: {\n    docs: {\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          'packageDir': { 'type': ['string', 'array'] },\n        },\n        'additionalProperties': false,\n      },\n    ],\n  },\n\n  create: function (context) {\n    const options = context.options[0] || {}\n    const filename = context.getFilename()\n    const deps = getDependencies(context, options.packageDir)\n\n    if (!deps) {\n      return {}\n    }\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    }\n\n    // todo: use module visitor from module-utils core\n    return {\n      ImportDeclaration: function (node) {\n        reportIfMissing(context, deps, depsOptions, node, node.source.value)\n      },\n      CallExpression: function handleRequires(node) {\n        if (isStaticRequire(node)) {\n          reportIfMissing(context, deps, depsOptions, node, node.arguments[0].value)\n        }\n      },\n    }\n  },\n}\n"]}