no-restricted-modules.js 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178
  1. /**
  2. * @fileoverview Restrict usage of specified node modules.
  3. * @author Christian Schulz
  4. */
  5. "use strict";
  6. //------------------------------------------------------------------------------
  7. // Helpers
  8. //------------------------------------------------------------------------------
  9. const DEFAULT_MESSAGE_TEMPLATE = "'{{moduleName}}' module is restricted from being used.";
  10. const CUSTOM_MESSAGE_TEMPLATE = "'{{moduleName}}' module is restricted from being used. {{customMessage}}";
  11. //------------------------------------------------------------------------------
  12. // Rule Definition
  13. //------------------------------------------------------------------------------
  14. const ignore = require("ignore");
  15. const arrayOfStrings = {
  16. type: "array",
  17. items: { type: "string" },
  18. uniqueItems: true
  19. };
  20. const arrayOfStringsOrObjects = {
  21. type: "array",
  22. items: {
  23. anyOf: [
  24. { type: "string" },
  25. {
  26. type: "object",
  27. properties: {
  28. name: { type: "string" },
  29. message: {
  30. type: "string",
  31. minLength: 1
  32. }
  33. },
  34. additionalProperties: false,
  35. required: ["name"]
  36. }
  37. ]
  38. },
  39. uniqueItems: true
  40. };
  41. module.exports = {
  42. meta: {
  43. docs: {
  44. description: "disallow specified modules when loaded by `require`",
  45. category: "Node.js and CommonJS",
  46. recommended: false,
  47. url: "https://eslint.org/docs/rules/no-restricted-modules"
  48. },
  49. schema: {
  50. anyOf: [
  51. arrayOfStringsOrObjects,
  52. {
  53. type: "array",
  54. items: {
  55. type: "object",
  56. properties: {
  57. paths: arrayOfStringsOrObjects,
  58. patterns: arrayOfStrings
  59. },
  60. additionalProperties: false
  61. },
  62. additionalItems: false
  63. }
  64. ]
  65. }
  66. },
  67. create(context) {
  68. const options = Array.isArray(context.options) ? context.options : [];
  69. const isPathAndPatternsObject =
  70. typeof options[0] === "object" &&
  71. (options[0].hasOwnProperty("paths") || options[0].hasOwnProperty("patterns"));
  72. const restrictedPaths = (isPathAndPatternsObject ? options[0].paths : context.options) || [];
  73. const restrictedPatterns = (isPathAndPatternsObject ? options[0].patterns : []) || [];
  74. const restrictedPathMessages = restrictedPaths.reduce((memo, importName) => {
  75. if (typeof importName === "string") {
  76. memo[importName] = null;
  77. } else {
  78. memo[importName.name] = importName.message;
  79. }
  80. return memo;
  81. }, {});
  82. // if no imports are restricted we don"t need to check
  83. if (Object.keys(restrictedPaths).length === 0 && restrictedPatterns.length === 0) {
  84. return {};
  85. }
  86. const ig = ignore().add(restrictedPatterns);
  87. /**
  88. * Function to check if a node is a string literal.
  89. * @param {ASTNode} node The node to check.
  90. * @returns {boolean} If the node is a string literal.
  91. */
  92. function isString(node) {
  93. return node && node.type === "Literal" && typeof node.value === "string";
  94. }
  95. /**
  96. * Function to check if a node is a require call.
  97. * @param {ASTNode} node The node to check.
  98. * @returns {boolean} If the node is a require call.
  99. */
  100. function isRequireCall(node) {
  101. return node.callee.type === "Identifier" && node.callee.name === "require";
  102. }
  103. /**
  104. * Report a restricted path.
  105. * @param {node} node representing the restricted path reference
  106. * @returns {void}
  107. * @private
  108. */
  109. function reportPath(node) {
  110. const moduleName = node.arguments[0].value.trim();
  111. const customMessage = restrictedPathMessages[moduleName];
  112. const message = customMessage
  113. ? CUSTOM_MESSAGE_TEMPLATE
  114. : DEFAULT_MESSAGE_TEMPLATE;
  115. context.report({
  116. node,
  117. message,
  118. data: {
  119. moduleName,
  120. customMessage
  121. }
  122. });
  123. }
  124. /**
  125. * Check if the given name is a restricted path name
  126. * @param {string} name name of a variable
  127. * @returns {boolean} whether the variable is a restricted path or not
  128. * @private
  129. */
  130. function isRestrictedPath(name) {
  131. return Object.prototype.hasOwnProperty.call(restrictedPathMessages, name);
  132. }
  133. return {
  134. CallExpression(node) {
  135. if (isRequireCall(node)) {
  136. // node has arguments and first argument is string
  137. if (node.arguments.length && isString(node.arguments[0])) {
  138. const moduleName = node.arguments[0].value.trim();
  139. // check if argument value is in restricted modules array
  140. if (isRestrictedPath(moduleName)) {
  141. reportPath(node);
  142. }
  143. if (restrictedPatterns.length > 0 && ig.ignores(moduleName)) {
  144. context.report({
  145. node,
  146. message: "'{{moduleName}}' module is restricted from being used by a pattern.",
  147. data: { moduleName }
  148. });
  149. }
  150. }
  151. }
  152. }
  153. };
  154. }
  155. };