newline-after-import.js 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249
  1. 'use strict';
  2. var _contextCompat = require('eslint-module-utils/contextCompat');
  3. var _staticRequire = require('../core/staticRequire');var _staticRequire2 = _interopRequireDefault(_staticRequire);
  4. var _docsUrl = require('../docsUrl');var _docsUrl2 = _interopRequireDefault(_docsUrl);
  5. var _debug = require('debug');var _debug2 = _interopRequireDefault(_debug);function _interopRequireDefault(obj) {return obj && obj.__esModule ? obj : { 'default': obj };} /**
  6. * @fileoverview Rule to enforce new line after import not followed by another import.
  7. * @author Radek Benkel
  8. */var log = (0, _debug2['default'])('eslint-plugin-import:rules:newline-after-import'); //------------------------------------------------------------------------------
  9. // Rule Definition
  10. //------------------------------------------------------------------------------
  11. function containsNodeOrEqual(outerNode, innerNode) {
  12. return outerNode.range[0] <= innerNode.range[0] && outerNode.range[1] >= innerNode.range[1];
  13. }
  14. function getScopeBody(scope) {
  15. if (scope.block.type === 'SwitchStatement') {
  16. log('SwitchStatement scopes not supported');
  17. return null;
  18. }var
  19. body = scope.block.body;
  20. if (body && body.type === 'BlockStatement') {
  21. return body.body;
  22. }
  23. return body;
  24. }
  25. function findNodeIndexInScopeBody(body, nodeToFind) {
  26. return body.findIndex(function (node) {return containsNodeOrEqual(node, nodeToFind);});
  27. }
  28. function getLineDifference(node, nextNode) {
  29. return nextNode.loc.start.line - node.loc.end.line;
  30. }
  31. function isClassWithDecorator(node) {
  32. return node.type === 'ClassDeclaration' && node.decorators && node.decorators.length;
  33. }
  34. function isExportDefaultClass(node) {
  35. return node.type === 'ExportDefaultDeclaration' && node.declaration.type === 'ClassDeclaration';
  36. }
  37. function isExportNameClass(node) {
  38. return node.type === 'ExportNamedDeclaration' && node.declaration && node.declaration.type === 'ClassDeclaration';
  39. }
  40. module.exports = {
  41. meta: {
  42. type: 'layout',
  43. docs: {
  44. category: 'Style guide',
  45. description: 'Enforce a newline after import statements.',
  46. url: (0, _docsUrl2['default'])('newline-after-import') },
  47. fixable: 'whitespace',
  48. schema: [
  49. {
  50. type: 'object',
  51. properties: {
  52. count: {
  53. type: 'integer',
  54. minimum: 1 },
  55. exactCount: { type: 'boolean' },
  56. considerComments: { type: 'boolean' } },
  57. additionalProperties: false }] },
  58. create: function () {function create(context) {
  59. var level = 0;
  60. var requireCalls = [];
  61. var options = Object.assign({
  62. count: 1,
  63. exactCount: false,
  64. considerComments: false },
  65. context.options[0]);
  66. function checkForNewLine(node, nextNode, type) {
  67. if (isExportDefaultClass(nextNode) || isExportNameClass(nextNode)) {
  68. var classNode = nextNode.declaration;
  69. if (isClassWithDecorator(classNode)) {
  70. nextNode = classNode.decorators[0];
  71. }
  72. } else if (isClassWithDecorator(nextNode)) {
  73. nextNode = nextNode.decorators[0];
  74. }
  75. var lineDifference = getLineDifference(node, nextNode);
  76. var EXPECTED_LINE_DIFFERENCE = options.count + 1;
  77. if (
  78. lineDifference < EXPECTED_LINE_DIFFERENCE ||
  79. options.exactCount && lineDifference !== EXPECTED_LINE_DIFFERENCE)
  80. {
  81. var column = node.loc.start.column;
  82. if (node.loc.start.line !== node.loc.end.line) {
  83. column = 0;
  84. }
  85. context.report({
  86. loc: {
  87. line: node.loc.end.line,
  88. column: column },
  89. message: 'Expected ' + String(options.count) + ' empty line' + (options.count > 1 ? 's' : '') + ' after ' + String(type) + ' statement not followed by another ' + String(type) + '.',
  90. fix: options.exactCount && EXPECTED_LINE_DIFFERENCE < lineDifference ? undefined : function (fixer) {return fixer.insertTextAfter(
  91. node,
  92. '\n'.repeat(EXPECTED_LINE_DIFFERENCE - lineDifference));} });
  93. }
  94. }
  95. function commentAfterImport(node, nextComment, type) {
  96. var lineDifference = getLineDifference(node, nextComment);
  97. var EXPECTED_LINE_DIFFERENCE = options.count + 1;
  98. if (lineDifference < EXPECTED_LINE_DIFFERENCE) {
  99. var column = node.loc.start.column;
  100. if (node.loc.start.line !== node.loc.end.line) {
  101. column = 0;
  102. }
  103. context.report({
  104. loc: {
  105. line: node.loc.end.line,
  106. column: column },
  107. message: 'Expected ' + String(options.count) + ' empty line' + (options.count > 1 ? 's' : '') + ' after ' + String(type) + ' statement not followed by another ' + String(type) + '.',
  108. fix: options.exactCount && EXPECTED_LINE_DIFFERENCE < lineDifference ? undefined : function (fixer) {return fixer.insertTextAfter(
  109. node,
  110. '\n'.repeat(EXPECTED_LINE_DIFFERENCE - lineDifference));} });
  111. }
  112. }
  113. function incrementLevel() {
  114. level++;
  115. }
  116. function decrementLevel() {
  117. level--;
  118. }
  119. function checkImport(node) {var
  120. parent = node.parent;
  121. if (!parent || !parent.body) {
  122. return;
  123. }
  124. var nodePosition = parent.body.indexOf(node);
  125. var nextNode = parent.body[nodePosition + 1];
  126. var endLine = node.loc.end.line;
  127. var nextComment = void 0;
  128. if (typeof parent.comments !== 'undefined' && options.considerComments) {
  129. nextComment = parent.comments.find(function (o) {return o.loc.start.line >= endLine && o.loc.start.line <= endLine + options.count + 1;});
  130. }
  131. // skip "export import"s
  132. if (node.type === 'TSImportEqualsDeclaration' && node.isExport) {
  133. return;
  134. }
  135. if (nextComment && typeof nextComment !== 'undefined') {
  136. commentAfterImport(node, nextComment, 'import');
  137. } else if (nextNode && nextNode.type !== 'ImportDeclaration' && (nextNode.type !== 'TSImportEqualsDeclaration' || nextNode.isExport)) {
  138. checkForNewLine(node, nextNode, 'import');
  139. }
  140. }
  141. return {
  142. ImportDeclaration: checkImport,
  143. TSImportEqualsDeclaration: checkImport,
  144. CallExpression: function () {function CallExpression(node) {
  145. if ((0, _staticRequire2['default'])(node) && level === 0) {
  146. requireCalls.push(node);
  147. }
  148. }return CallExpression;}(),
  149. 'Program:exit': function () {function ProgramExit(node) {
  150. log('exit processing for', (0, _contextCompat.getPhysicalFilename)(context));
  151. var scopeBody = getScopeBody((0, _contextCompat.getScope)(context, node));
  152. log('got scope:', scopeBody);
  153. requireCalls.forEach(function (node, index) {
  154. var nodePosition = findNodeIndexInScopeBody(scopeBody, node);
  155. log('node position in scope:', nodePosition);
  156. var statementWithRequireCall = scopeBody[nodePosition];
  157. var nextStatement = scopeBody[nodePosition + 1];
  158. var nextRequireCall = requireCalls[index + 1];
  159. if (nextRequireCall && containsNodeOrEqual(statementWithRequireCall, nextRequireCall)) {
  160. return;
  161. }
  162. if (
  163. nextStatement && (
  164. !nextRequireCall ||
  165. !containsNodeOrEqual(nextStatement, nextRequireCall)))
  166. {
  167. var nextComment = void 0;
  168. if (typeof statementWithRequireCall.parent.comments !== 'undefined' && options.considerComments) {
  169. var endLine = node.loc.end.line;
  170. nextComment = statementWithRequireCall.parent.comments.find(function (o) {return o.loc.start.line >= endLine && o.loc.start.line <= endLine + options.count + 1;});
  171. }
  172. if (nextComment && typeof nextComment !== 'undefined') {
  173. commentAfterImport(statementWithRequireCall, nextComment, 'require');
  174. } else {
  175. checkForNewLine(statementWithRequireCall, nextStatement, 'require');
  176. }
  177. }
  178. });
  179. }return ProgramExit;}(),
  180. FunctionDeclaration: incrementLevel,
  181. FunctionExpression: incrementLevel,
  182. ArrowFunctionExpression: incrementLevel,
  183. BlockStatement: incrementLevel,
  184. ObjectExpression: incrementLevel,
  185. Decorator: incrementLevel,
  186. 'FunctionDeclaration:exit': decrementLevel,
  187. 'FunctionExpression:exit': decrementLevel,
  188. 'ArrowFunctionExpression:exit': decrementLevel,
  189. 'BlockStatement:exit': decrementLevel,
  190. 'ObjectExpression:exit': decrementLevel,
  191. 'Decorator:exit': decrementLevel };
  192. }return create;}() };
  193. //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/rules/newline-after-import.js"],"names":["log","containsNodeOrEqual","outerNode","innerNode","range","getScopeBody","scope","block","type","body","findNodeIndexInScopeBody","nodeToFind","findIndex","node","getLineDifference","nextNode","loc","start","line","end","isClassWithDecorator","decorators","length","isExportDefaultClass","declaration","isExportNameClass","module","exports","meta","docs","category","description","url","fixable","schema","properties","count","minimum","exactCount","considerComments","additionalProperties","create","context","level","requireCalls","options","checkForNewLine","classNode","lineDifference","EXPECTED_LINE_DIFFERENCE","column","report","message","fix","undefined","fixer","insertTextAfter","repeat","commentAfterImport","nextComment","incrementLevel","decrementLevel","checkImport","parent","nodePosition","indexOf","endLine","comments","find","o","isExport","ImportDeclaration","TSImportEqualsDeclaration","CallExpression","push","scopeBody","forEach","index","statementWithRequireCall","nextStatement","nextRequireCall","FunctionDeclaration","FunctionExpression","ArrowFunctionExpression","BlockStatement","ObjectExpression","Decorator"],"mappings":"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","file":"newline-after-import.js","sourcesContent":["/**\n * @fileoverview Rule to enforce new line after import not followed by another import.\n * @author Radek Benkel\n */\n\nimport { getPhysicalFilename, getScope } from 'eslint-module-utils/contextCompat';\n\nimport isStaticRequire from '../core/staticRequire';\nimport docsUrl from '../docsUrl';\n\nimport debug from 'debug';\nconst log = debug('eslint-plugin-import:rules:newline-after-import');\n\n//------------------------------------------------------------------------------\n// Rule Definition\n//------------------------------------------------------------------------------\n\nfunction containsNodeOrEqual(outerNode, innerNode) {\n  return outerNode.range[0] <= innerNode.range[0] && outerNode.range[1] >= innerNode.range[1];\n}\n\nfunction getScopeBody(scope) {\n  if (scope.block.type === 'SwitchStatement') {\n    log('SwitchStatement scopes not supported');\n    return null;\n  }\n\n  const { body } = scope.block;\n  if (body && body.type === 'BlockStatement') {\n    return body.body;\n  }\n\n  return body;\n}\n\nfunction findNodeIndexInScopeBody(body, nodeToFind) {\n  return body.findIndex((node) => containsNodeOrEqual(node, nodeToFind));\n}\n\nfunction getLineDifference(node, nextNode) {\n  return nextNode.loc.start.line - node.loc.end.line;\n}\n\nfunction isClassWithDecorator(node) {\n  return node.type === 'ClassDeclaration' && node.decorators && node.decorators.length;\n}\n\nfunction isExportDefaultClass(node) {\n  return node.type === 'ExportDefaultDeclaration' && node.declaration.type === 'ClassDeclaration';\n}\n\nfunction isExportNameClass(node) {\n\n  return node.type === 'ExportNamedDeclaration' && node.declaration && node.declaration.type === 'ClassDeclaration';\n}\n\nmodule.exports = {\n  meta: {\n    type: 'layout',\n    docs: {\n      category: 'Style guide',\n      description: 'Enforce a newline after import statements.',\n      url: docsUrl('newline-after-import'),\n    },\n    fixable: 'whitespace',\n    schema: [\n      {\n        type: 'object',\n        properties: {\n          count: {\n            type: 'integer',\n            minimum: 1,\n          },\n          exactCount: { type: 'boolean' },\n          considerComments: { type: 'boolean' },\n        },\n        additionalProperties: false,\n      },\n    ],\n  },\n  create(context) {\n    let level = 0;\n    const requireCalls = [];\n    const options = {\n      count: 1,\n      exactCount: false,\n      considerComments: false,\n      ...context.options[0],\n    };\n\n    function checkForNewLine(node, nextNode, type) {\n      if (isExportDefaultClass(nextNode) || isExportNameClass(nextNode)) {\n        const classNode = nextNode.declaration;\n\n        if (isClassWithDecorator(classNode)) {\n          nextNode = classNode.decorators[0];\n        }\n      } else if (isClassWithDecorator(nextNode)) {\n        nextNode = nextNode.decorators[0];\n      }\n\n      const lineDifference = getLineDifference(node, nextNode);\n      const EXPECTED_LINE_DIFFERENCE = options.count + 1;\n\n      if (\n        lineDifference < EXPECTED_LINE_DIFFERENCE\n        || options.exactCount && lineDifference !== EXPECTED_LINE_DIFFERENCE\n      ) {\n        let column = node.loc.start.column;\n\n        if (node.loc.start.line !== node.loc.end.line) {\n          column = 0;\n        }\n\n        context.report({\n          loc: {\n            line: node.loc.end.line,\n            column,\n          },\n          message: `Expected ${options.count} empty line${options.count > 1 ? 's' : ''} after ${type} statement not followed by another ${type}.`,\n          fix: options.exactCount && EXPECTED_LINE_DIFFERENCE < lineDifference ? undefined : (fixer) => fixer.insertTextAfter(\n            node,\n            '\\n'.repeat(EXPECTED_LINE_DIFFERENCE - lineDifference),\n          ),\n        });\n      }\n    }\n\n    function commentAfterImport(node, nextComment, type) {\n      const lineDifference = getLineDifference(node, nextComment);\n      const EXPECTED_LINE_DIFFERENCE = options.count + 1;\n\n      if (lineDifference < EXPECTED_LINE_DIFFERENCE) {\n        let column = node.loc.start.column;\n\n        if (node.loc.start.line !== node.loc.end.line) {\n          column = 0;\n        }\n\n        context.report({\n          loc: {\n            line: node.loc.end.line,\n            column,\n          },\n          message: `Expected ${options.count} empty line${options.count > 1 ? 's' : ''} after ${type} statement not followed by another ${type}.`,\n          fix: options.exactCount && EXPECTED_LINE_DIFFERENCE < lineDifference ? undefined : (fixer) => fixer.insertTextAfter(\n            node,\n            '\\n'.repeat(EXPECTED_LINE_DIFFERENCE - lineDifference),\n          ),\n        });\n      }\n    }\n\n    function incrementLevel() {\n      level++;\n    }\n    function decrementLevel() {\n      level--;\n    }\n\n    function checkImport(node) {\n      const { parent } = node;\n\n      if (!parent || !parent.body) {\n        return;\n      }\n\n      const nodePosition = parent.body.indexOf(node);\n      const nextNode = parent.body[nodePosition + 1];\n      const endLine = node.loc.end.line;\n      let nextComment;\n\n      if (typeof parent.comments !== 'undefined' && options.considerComments) {\n        nextComment = parent.comments.find((o) => o.loc.start.line >= endLine && o.loc.start.line <= endLine + options.count + 1);\n      }\n\n      // skip \"export import\"s\n      if (node.type === 'TSImportEqualsDeclaration' && node.isExport) {\n        return;\n      }\n\n      if (nextComment && typeof nextComment !== 'undefined') {\n        commentAfterImport(node, nextComment, 'import');\n      } else if (nextNode && nextNode.type !== 'ImportDeclaration' && (nextNode.type !== 'TSImportEqualsDeclaration' || nextNode.isExport)) {\n        checkForNewLine(node, nextNode, 'import');\n      }\n    }\n\n    return {\n      ImportDeclaration: checkImport,\n      TSImportEqualsDeclaration: checkImport,\n      CallExpression(node) {\n        if (isStaticRequire(node) && level === 0) {\n          requireCalls.push(node);\n        }\n      },\n      'Program:exit'(node) {\n        log('exit processing for', getPhysicalFilename(context));\n        const scopeBody = getScopeBody(getScope(context, node));\n        log('got scope:', scopeBody);\n\n        requireCalls.forEach((node, index) => {\n          const nodePosition = findNodeIndexInScopeBody(scopeBody, node);\n          log('node position in scope:', nodePosition);\n\n          const statementWithRequireCall = scopeBody[nodePosition];\n          const nextStatement = scopeBody[nodePosition + 1];\n          const nextRequireCall = requireCalls[index + 1];\n\n          if (nextRequireCall && containsNodeOrEqual(statementWithRequireCall, nextRequireCall)) {\n            return;\n          }\n\n          if (\n            nextStatement && (\n              !nextRequireCall\n              || !containsNodeOrEqual(nextStatement, nextRequireCall)\n            )\n          ) {\n            let nextComment;\n            if (typeof statementWithRequireCall.parent.comments !== 'undefined' && options.considerComments) {\n              const endLine = node.loc.end.line;\n              nextComment = statementWithRequireCall.parent.comments.find((o) => o.loc.start.line >= endLine && o.loc.start.line <= endLine + options.count + 1);\n            }\n\n            if (nextComment && typeof nextComment !== 'undefined') {\n\n              commentAfterImport(statementWithRequireCall, nextComment, 'require');\n            } else {\n              checkForNewLine(statementWithRequireCall, nextStatement, 'require');\n            }\n          }\n        });\n      },\n      FunctionDeclaration: incrementLevel,\n      FunctionExpression: incrementLevel,\n      ArrowFunctionExpression: incrementLevel,\n      BlockStatement: incrementLevel,\n      ObjectExpression: incrementLevel,\n      Decorator: incrementLevel,\n      'FunctionDeclaration:exit': decrementLevel,\n      'FunctionExpression:exit': decrementLevel,\n      'ArrowFunctionExpression:exit': decrementLevel,\n      'BlockStatement:exit': decrementLevel,\n      'ObjectExpression:exit': decrementLevel,\n      'Decorator:exit': decrementLevel,\n    };\n  },\n};\n"]}