diff --git a/src/rules/order.js b/src/rules/order.js index 2f4ef08b70..bf8cbfc590 100644 --- a/src/rules/order.js +++ b/src/rules/order.js @@ -336,7 +336,7 @@ function registerNode(context, importEntry, ranks, imported, excludedImportTypes } } -function isModuleLevelRequire(node) { +function getRequireBlock(node) { let n = node; // Handle cases like `const baz = require('foo').bar.baz` // and `const foo = require('foo')()` @@ -346,11 +346,13 @@ function isModuleLevelRequire(node) { ) { n = n.parent; } - return ( + if ( n.parent.type === 'VariableDeclarator' && n.parent.parent.type === 'VariableDeclaration' && n.parent.parent.parent.type === 'Program' - ); + ) { + return n.parent.parent.parent; + } } const types = ['builtin', 'external', 'internal', 'unknown', 'parent', 'sibling', 'index', 'object', 'type']; @@ -605,7 +607,14 @@ module.exports = { }, }; } - let imported = []; + const importMap = new Map(); + + function getBlockImports(node) { + if (!importMap.has(node)) { + importMap.set(node, []); + } + return importMap.get(node); + } return { ImportDeclaration: function handleImports(node) { @@ -621,7 +630,7 @@ module.exports = { type: 'import', }, ranks, - imported, + getBlockImports(node.parent), pathGroupsExcludedImportTypes ); } @@ -652,12 +661,16 @@ module.exports = { type, }, ranks, - imported, + getBlockImports(node.parent), pathGroupsExcludedImportTypes ); }, CallExpression: function handleRequires(node) { - if (!isStaticRequire(node) || !isModuleLevelRequire(node)) { + if (!isStaticRequire(node)) { + return; + } + const block = getRequireBlock(node); + if (!block) { return; } const name = node.arguments[0].value; @@ -670,22 +683,24 @@ module.exports = { type: 'require', }, ranks, - imported, + getBlockImports(block), pathGroupsExcludedImportTypes ); }, - 'Program:exit': function reportAndReset() { - if (newlinesBetweenImports !== 'ignore') { - makeNewlinesBetweenReport(context, imported, newlinesBetweenImports); - } + 'Program:exit': function reportAndReset(node) { + for (const imported of importMap.values()) { - if (alphabetize.order !== 'ignore') { - mutateRanksToAlphabetize(imported, alphabetize); - } + if (newlinesBetweenImports !== 'ignore') { + makeNewlinesBetweenReport(context, imported, newlinesBetweenImports); + } - makeOutOfOrderReport(context, imported); + if (alphabetize.order !== 'ignore') { + mutateRanksToAlphabetize(imported, alphabetize); + } - imported = []; + makeOutOfOrderReport(context, imported); + } + importMap.clear(); }, }; }, diff --git a/tests/src/rules/order.js b/tests/src/rules/order.js index 61bf1bb70f..146306259f 100644 --- a/tests/src/rules/order.js +++ b/tests/src/rules/order.js @@ -2462,6 +2462,24 @@ context('TypeScript', function () { }, ], }), + // Imports inside module declaration + test({ + code: ` + import type { CopyOptions } from 'fs'; + import type { ParsedPath } from 'path'; + + declare module 'my-module' { + import type { CopyOptions } from 'fs'; + import type { ParsedPath } from 'path'; + } + `, + ...parserConfig, + options: [ + { + alphabetize: { order: 'asc' }, + }, + ], + }), ], invalid: [ // Option alphabetize: {order: 'asc'} @@ -2655,6 +2673,38 @@ context('TypeScript', function () { }], options: [{ warnOnUnassignedImports: true }], }), + // Imports inside module declaration + test({ + code: ` + import type { ParsedPath } from 'path'; + import type { CopyOptions } from 'fs'; + + declare module 'my-module' { + import type { ParsedPath } from 'path'; + import type { CopyOptions } from 'fs'; + } + `, + output: ` + import type { CopyOptions } from 'fs'; + import type { ParsedPath } from 'path'; + + declare module 'my-module' { + import type { CopyOptions } from 'fs'; + import type { ParsedPath } from 'path'; + } + `, + errors: [{ + message: '`fs` import should occur before import of `path`', + },{ + message: '`fs` import should occur before import of `path`', + }], + ...parserConfig, + options: [ + { + alphabetize: { order: 'asc' }, + }, + ], + }), ], }); });