'use strict'; var _path = require('path'); var _path2 = _interopRequireDefault(_path); var _fs = require('fs'); var _fs2 = _interopRequireDefault(_fs); var _readPkgUp = require('read-pkg-up'); var _readPkgUp2 = _interopRequireDefault(_readPkgUp); var _minimatch = require('minimatch'); var _minimatch2 = _interopRequireDefault(_minimatch); var _importType = require('../core/importType'); var _importType2 = _interopRequireDefault(_importType); var _staticRequire = require('../core/staticRequire'); var _staticRequire2 = _interopRequireDefault(_staticRequire); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function getDependencies(context, packageDir) { try { const packageContent = packageDir ? JSON.parse(_fs2.default.readFileSync(_path2.default.join(packageDir, 'package.json'), 'utf8')) : _readPkgUp2.default.sync({ cwd: context.getFilename(), normalize: false }).pkg; if (!packageContent) { return null; } return { dependencies: packageContent.dependencies || {}, devDependencies: packageContent.devDependencies || {}, optionalDependencies: packageContent.optionalDependencies || {}, peerDependencies: packageContent.peerDependencies || {} }; } catch (e) { if (packageDir && e.code === 'ENOENT') { context.report({ message: 'The package.json file could not be found.', loc: { line: 0, column: 0 } }); } if (e.name === 'JSONError' || e instanceof SyntaxError) { context.report({ message: 'The package.json file could not be parsed: ' + e.message, loc: { line: 0, column: 0 } }); } return null; } } function missingErrorMessage(packageName) { return `'${packageName}' should be listed in the project's dependencies. ` + `Run 'npm i -S ${packageName}' to add it`; } function devDepErrorMessage(packageName) { return `'${packageName}' should be listed in the project's dependencies, not devDependencies.`; } function optDepErrorMessage(packageName) { return `'${packageName}' should be listed in the project's dependencies, ` + `not optionalDependencies.`; } function reportIfMissing(context, deps, depsOptions, node, name) { if ((0, _importType2.default)(name, context) !== 'external') { return; } const splitName = name.split('/'); const packageName = splitName[0][0] === '@' ? splitName.slice(0, 2).join('/') : splitName[0]; const isInDeps = deps.dependencies[packageName] !== undefined; const isInDevDeps = deps.devDependencies[packageName] !== undefined; const isInOptDeps = deps.optionalDependencies[packageName] !== undefined; const isInPeerDeps = deps.peerDependencies[packageName] !== undefined; if (isInDeps || depsOptions.allowDevDeps && isInDevDeps || depsOptions.allowPeerDeps && isInPeerDeps || depsOptions.allowOptDeps && isInOptDeps) { return; } if (isInDevDeps && !depsOptions.allowDevDeps) { context.report(node, devDepErrorMessage(packageName)); return; } if (isInOptDeps && !depsOptions.allowOptDeps) { context.report(node, optDepErrorMessage(packageName)); return; } context.report(node, missingErrorMessage(packageName)); } function testConfig(config, filename) { // Simplest configuration first, either a boolean or nothing. if (typeof config === 'boolean' || typeof config === 'undefined') { return config; } // Array of globs. return config.some(c => (0, _minimatch2.default)(filename, c) || (0, _minimatch2.default)(filename, _path2.default.join(process.cwd(), c))); } module.exports = { meta: { docs: {}, schema: [{ 'type': 'object', 'properties': { 'devDependencies': { 'type': ['boolean', 'array'] }, 'optionalDependencies': { 'type': ['boolean', 'array'] }, 'peerDependencies': { 'type': ['boolean', 'array'] }, 'packageDir': { 'type': 'string' } }, 'additionalProperties': false }] }, create: function (context) { const options = context.options[0] || {}; const filename = context.getFilename(); const deps = getDependencies(context, options.packageDir); if (!deps) { return {}; } const depsOptions = { allowDevDeps: testConfig(options.devDependencies, filename) !== false, allowOptDeps: testConfig(options.optionalDependencies, filename) !== false, allowPeerDeps: testConfig(options.peerDependencies, filename) !== false }; // todo: use module visitor from module-utils core return { ImportDeclaration: function (node) { reportIfMissing(context, deps, depsOptions, node, node.source.value); }, CallExpression: function handleRequires(node) { if ((0, _staticRequire2.default)(node)) { reportIfMissing(context, deps, depsOptions, node, node.arguments[0].value); } } }; } }; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["rules/no-extraneous-dependencies.js"],"names":["getDependencies","context","packageDir","packageContent","JSON","parse","readFileSync","join","sync","cwd","getFilename","normalize","pkg","dependencies","devDependencies","optionalDependencies","peerDependencies","e","code","report","message","loc","line","column","name","SyntaxError","missingErrorMessage","packageName","devDepErrorMessage","optDepErrorMessage","reportIfMissing","deps","depsOptions","node","splitName","split","slice","isInDeps","undefined","isInDevDeps","isInOptDeps","isInPeerDeps","allowDevDeps","allowPeerDeps","allowOptDeps","testConfig","config","filename","some","c","process","module","exports","meta","docs","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 readPkgUp from 'read-pkg-up'\nimport minimatch from 'minimatch'\nimport importType from '../core/importType'\nimport isStaticRequire from '../core/staticRequire'\n\nfunction getDependencies(context, packageDir) {\n  try {\n    const packageContent = packageDir\n      ? JSON.parse(fs.readFileSync(path.join(packageDir, 'package.json'), 'utf8'))\n      : readPkgUp.sync({cwd: context.getFilename(), normalize: false}).pkg\n\n    if (!packageContent) {\n      return null\n    }\n\n    return {\n      dependencies: packageContent.dependencies || {},\n      devDependencies: packageContent.devDependencies || {},\n      optionalDependencies: packageContent.optionalDependencies || {},\n      peerDependencies: packageContent.peerDependencies || {},\n    }\n  } catch (e) {\n    if (packageDir && 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  if (importType(name, context) !== 'external') {\n    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\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' },\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"]}