@borela-tech/eslint-config 2.1.0 → 2.1.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -38,21 +38,21 @@ function categorizeStatements(statements) {
38
38
  return result;
39
39
  }
40
40
 
41
- // src/rules/importsAndReExportsAtTop/findFirstIndices.ts
42
- function findFirstIndices(statements) {
43
- let firstImport = Infinity;
44
- let firstReExport = Infinity;
45
- let firstOther = -1;
41
+ // src/rules/importsAndReExportsAtTop/findStatementIndices.ts
42
+ function findStatementIndices(statements) {
43
+ let firstRegularStatement = -1;
44
+ let lastImport = -1;
45
+ let lastReExport = -1;
46
46
  for (let i = 0; i < statements.length; i++) {
47
47
  const type = getStatementType(statements[i]);
48
- if (type === "import" && firstImport === Infinity)
49
- firstImport = i;
50
- else if (type === "re-export" && firstReExport === Infinity)
51
- firstReExport = i;
52
- else if (type === "other" && firstOther === -1)
53
- firstOther = i;
48
+ if (type === "import")
49
+ lastImport = i;
50
+ else if (type === "re-export")
51
+ lastReExport = i;
52
+ else if (type === "other" && firstRegularStatement === -1)
53
+ firstRegularStatement = i;
54
54
  }
55
- return { firstImport, firstReExport, firstOther };
55
+ return { firstRegularStatement, lastImport, lastReExport };
56
56
  }
57
57
 
58
58
  // src/rules/importsAndReExportsAtTop/generateSortedText.ts
@@ -69,18 +69,12 @@ function generateSortedText(context, categories) {
69
69
 
70
70
  // src/rules/importsAndReExportsAtTop/hasViolation.ts
71
71
  function hasViolation(indices, categories) {
72
- const {
73
- firstImport,
74
- firstReExport,
75
- firstOther
76
- } = indices;
77
- if (categories.imports.length === 0 || categories.reExports.length === 0)
72
+ const { firstRegularStatement, lastImport, lastReExport } = indices;
73
+ if (categories.imports.length === 0 && categories.reExports.length === 0)
78
74
  return false;
79
- const firstImportOrReExport = Math.min(firstImport, firstReExport);
80
- const hasOtherBeforeImportOrReExport = firstOther !== -1 && firstOther < firstImportOrReExport;
81
- if (hasOtherBeforeImportOrReExport || firstImport > firstReExport)
82
- return true;
83
- return false;
75
+ const hasImportAfterRegularStatement = categories.imports.length > 0 && firstRegularStatement !== -1 && lastImport > firstRegularStatement;
76
+ const hasReExportAfterRegularStatement = categories.reExports.length > 0 && firstRegularStatement !== -1 && lastReExport > firstRegularStatement;
77
+ return hasImportAfterRegularStatement || hasReExportAfterRegularStatement;
84
78
  }
85
79
 
86
80
  // src/rules/importsAndReExportsAtTop/index.ts
@@ -102,7 +96,7 @@ var importsAndReExportsAtTop = {
102
96
  Program(node) {
103
97
  const statements = node.body;
104
98
  const categories = categorizeStatements(statements);
105
- const indices = findFirstIndices(statements);
99
+ const indices = findStatementIndices(statements);
106
100
  if (!hasViolation(indices, categories))
107
101
  return;
108
102
  context.report({
@@ -345,28 +339,6 @@ function buildSortedCode(grouped, sourceCode) {
345
339
  return sortedCode;
346
340
  }
347
341
 
348
- // src/rules/sortedImports/createFix/findLastImportIndex.ts
349
- function findLastImportIndex(programBody) {
350
- let lastIndex = 0;
351
- for (let i = 0; i < programBody.length; i++) {
352
- if (programBody[i].type === "ImportDeclaration")
353
- lastIndex = i;
354
- else
355
- break;
356
- }
357
- return lastIndex;
358
- }
359
-
360
- // src/rules/sortedImports/createFix/getReplacementRange.ts
361
- function getReplacementRange(programBody) {
362
- const lastIndex = findLastImportIndex(programBody);
363
- const firstImport = programBody[0];
364
- const lastImport = programBody[lastIndex];
365
- const start = firstImport.range[0];
366
- const end = lastImport.range[1];
367
- return { start, end };
368
- }
369
-
370
342
  // src/rules/sortedImports/createFix/groupImportsByType.ts
371
343
  function groupImportsByType(categorized) {
372
344
  const grouped = {
@@ -389,23 +361,48 @@ function sortImportGroups(grouped) {
389
361
  }
390
362
 
391
363
  // src/rules/sortedImports/createFix/index.ts
392
- function createFix(fixer, importDeclarations, sourceCode, programBody) {
393
- const range = getReplacementRange(programBody);
364
+ function createFixForGroup(fixer, importDeclarations, sourceCode) {
365
+ if (importDeclarations.length === 0) {
366
+ return null;
367
+ }
394
368
  const categorized = categorizeImports(importDeclarations);
395
369
  const grouped = groupImportsByType(categorized);
396
370
  sortImportGroups(grouped);
397
371
  const sortedCode = buildSortedCode(grouped, sourceCode).join("\n");
372
+ const firstImport = importDeclarations[0];
373
+ const lastImport = importDeclarations[importDeclarations.length - 1];
398
374
  return fixer.replaceTextRange(
399
- [range.start, range.end],
375
+ [firstImport.range[0], lastImport.range[1]],
400
376
  sortedCode
401
377
  );
402
378
  }
379
+ function createFix(fixer, importGroups, sourceCode) {
380
+ const fixes = [];
381
+ for (const group of importGroups) {
382
+ const fix = createFixForGroup(fixer, group, sourceCode);
383
+ if (fix)
384
+ fixes.push(fix);
385
+ }
386
+ return fixes;
387
+ }
403
388
 
404
- // src/rules/sortedImports/getImportDeclarations.ts
405
- function getImportDeclarations(programBody) {
406
- return programBody.filter(
407
- (statement) => statement.type === "ImportDeclaration"
408
- );
389
+ // src/rules/sortedImports/getImportGroups.ts
390
+ function getImportGroups(programBody) {
391
+ const groups = [];
392
+ let currentGroup = [];
393
+ for (const statement of programBody) {
394
+ if (statement.type === "ImportDeclaration") {
395
+ currentGroup.push(statement);
396
+ continue;
397
+ }
398
+ if (currentGroup.length > 0) {
399
+ groups.push(currentGroup);
400
+ currentGroup = [];
401
+ }
402
+ }
403
+ if (currentGroup.length > 0)
404
+ groups.push(currentGroup);
405
+ return groups;
409
406
  }
410
407
 
411
408
  // src/rules/sortedImports/index.ts
@@ -428,22 +425,26 @@ var sortedImports = {
428
425
  return {
429
426
  Program(node) {
430
427
  const body = node.body;
431
- const declarations = getImportDeclarations(body);
432
- if (declarations.length === 0)
428
+ const importGroups = getImportGroups(body);
429
+ if (importGroups.length === 0)
433
430
  return;
434
- const categorized = categorizeImports(declarations);
435
- const errors = [
436
- ...checkGroupOrdering(categorized),
437
- ...checkAlphabeticalSorting(categorized),
438
- ...checkSpecifiersSorting(categorized)
439
- ];
440
- for (const error of errors) {
431
+ const allErrors = [];
432
+ for (const group of importGroups) {
433
+ const categorized = categorizeImports(group);
434
+ const errors = [
435
+ ...checkGroupOrdering(categorized),
436
+ ...checkAlphabeticalSorting(categorized),
437
+ ...checkSpecifiersSorting(categorized)
438
+ ];
439
+ allErrors.push(...errors);
440
+ }
441
+ for (const error of allErrors) {
441
442
  context.report({
442
443
  node: error.node,
443
444
  messageId: error.messageId,
444
445
  fix(fixer) {
445
446
  const sourceCode = context.sourceCode;
446
- return createFix(fixer, declarations, sourceCode, body);
447
+ return createFix(fixer, importGroups, sourceCode);
447
448
  }
448
449
  });
449
450
  }
@@ -608,40 +609,6 @@ function buildSortedCode2(grouped, sourceCode) {
608
609
  return sortedCode;
609
610
  }
610
611
 
611
- // src/rules/sortedReExports/createFix/findFirstExportIndex.ts
612
- function findFirstExportIndex(programBody) {
613
- for (let i = 0; i < programBody.length; i++) {
614
- if (programBody[i].type === "ExportNamedDeclaration" || programBody[i].type === "ExportAllDeclaration") {
615
- return i;
616
- }
617
- }
618
- return -1;
619
- }
620
-
621
- // src/rules/sortedReExports/createFix/findLastExportIndex.ts
622
- function findLastExportIndex(programBody) {
623
- let lastIndex = -1;
624
- for (let i = 0; i < programBody.length; i++) {
625
- if (programBody[i].type === "ExportNamedDeclaration" || programBody[i].type === "ExportAllDeclaration") {
626
- lastIndex = i;
627
- }
628
- }
629
- return lastIndex;
630
- }
631
-
632
- // src/rules/sortedReExports/createFix/getReplacementRange.ts
633
- function getReplacementRange2(programBody) {
634
- const firstIndex = findFirstExportIndex(programBody);
635
- const lastIndex = findLastExportIndex(programBody);
636
- if (firstIndex === -1 || lastIndex === -1)
637
- return { start: 0, end: 0 };
638
- const firstExport = programBody[firstIndex];
639
- const lastExport = programBody[lastIndex];
640
- const start = firstExport.range[0];
641
- const end = lastExport.range[1];
642
- return { start, end };
643
- }
644
-
645
612
  // src/rules/sortedReExports/createFix/groupReExportsByType.ts
646
613
  function groupReExportsByType(categorized) {
647
614
  const grouped = {
@@ -662,23 +629,51 @@ function sortExportGroups(grouped) {
662
629
  }
663
630
 
664
631
  // src/rules/sortedReExports/createFix/index.ts
665
- function createFix2(fixer, reExportDeclarations, sourceCode, programBody) {
666
- const range = getReplacementRange2(programBody);
632
+ function createFixForGroup2(fixer, reExportDeclarations, sourceCode) {
633
+ if (reExportDeclarations.length === 0) {
634
+ return null;
635
+ }
667
636
  const categorized = categorizeReExports(reExportDeclarations);
668
637
  const grouped = groupReExportsByType(categorized);
669
638
  sortExportGroups(grouped);
670
639
  const sortedCode = buildSortedCode2(grouped, sourceCode).join("\n");
640
+ const firstReExport = reExportDeclarations[0];
641
+ const lastReExport = reExportDeclarations[reExportDeclarations.length - 1];
671
642
  return fixer.replaceTextRange(
672
- [range.start, range.end],
643
+ [firstReExport.range[0], lastReExport.range[1]],
673
644
  sortedCode
674
645
  );
675
646
  }
647
+ function createFix2(fixer, reExportGroups, sourceCode) {
648
+ const fixes = [];
649
+ for (const group of reExportGroups) {
650
+ const fix = createFixForGroup2(fixer, group, sourceCode);
651
+ if (fix)
652
+ fixes.push(fix);
653
+ }
654
+ return fixes;
655
+ }
676
656
 
677
- // src/rules/sortedReExports/getReExportDeclarations.ts
678
- function getReExportDeclarations(programBody) {
679
- return programBody.filter(
680
- (statement) => statement.type === "ExportNamedDeclaration" && statement.source !== null || statement.type === "ExportAllDeclaration"
681
- );
657
+ // src/rules/sortedReExports/getReExportGroups.ts
658
+ function isReExportDeclaration(statement) {
659
+ return statement.type === "ExportNamedDeclaration" && statement.source !== null || statement.type === "ExportAllDeclaration";
660
+ }
661
+ function getReExportGroups(programBody) {
662
+ const groups = [];
663
+ let currentGroup = [];
664
+ for (const statement of programBody) {
665
+ if (isReExportDeclaration(statement)) {
666
+ currentGroup.push(statement);
667
+ continue;
668
+ }
669
+ if (currentGroup.length > 0) {
670
+ groups.push(currentGroup);
671
+ currentGroup = [];
672
+ }
673
+ }
674
+ if (currentGroup.length > 0)
675
+ groups.push(currentGroup);
676
+ return groups;
682
677
  }
683
678
 
684
679
  // src/rules/sortedReExports/index.ts
@@ -701,22 +696,26 @@ var sortedReExports = {
701
696
  return {
702
697
  Program(node) {
703
698
  const body = node.body;
704
- const declarations = getReExportDeclarations(body);
705
- if (declarations.length === 0)
699
+ const reExportGroups = getReExportGroups(body);
700
+ if (reExportGroups.length === 0)
706
701
  return;
707
- const categorized = categorizeReExports(declarations);
708
- const errors = [
709
- ...checkGroupOrdering2(categorized),
710
- ...checkAlphabeticalSorting2(categorized),
711
- ...checkSpecifiersSorting2(categorized)
712
- ];
713
- for (const error of errors) {
702
+ const allErrors = [];
703
+ for (const group of reExportGroups) {
704
+ const categorized = categorizeReExports(group);
705
+ const errors = [
706
+ ...checkGroupOrdering2(categorized),
707
+ ...checkAlphabeticalSorting2(categorized),
708
+ ...checkSpecifiersSorting2(categorized)
709
+ ];
710
+ allErrors.push(...errors);
711
+ }
712
+ for (const error of allErrors) {
714
713
  context.report({
715
714
  node: error.node,
716
715
  messageId: error.messageId,
717
716
  fix(fixer) {
718
717
  const sourceCode = context.sourceCode;
719
- return createFix2(fixer, declarations, sourceCode, body);
718
+ return createFix2(fixer, reExportGroups, sourceCode);
720
719
  }
721
720
  });
722
721
  }
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/rules/importsAndReExportsAtTop/getStatementType.ts","../src/rules/importsAndReExportsAtTop/categorizeStatements.ts","../src/rules/importsAndReExportsAtTop/findFirstIndices.ts","../src/rules/importsAndReExportsAtTop/generateSortedText.ts","../src/rules/importsAndReExportsAtTop/hasViolation.ts","../src/rules/importsAndReExportsAtTop/index.ts","../src/rules/individualImports.ts","../src/rules/individualReExports.ts","../src/rules/sortedImports/categorizeImport.ts","../src/rules/sortedImports/getSortKey.ts","../src/rules/sortedImports/categorizeImports.ts","../src/lib/compare.ts","../src/rules/sortedImports/ImportGroupOrder.ts","../src/rules/sortedImports/checkAlphabeticalSorting.ts","../src/rules/sortedImports/checkGroupOrdering.ts","../src/rules/sortedImports/getSpecifierName.ts","../src/rules/sortedImports/areSpecifiersSorted.ts","../src/rules/sortedImports/getNamedSpecifiers.ts","../src/rules/sortedImports/checkSpecifiersSorting.ts","../src/rules/sortedImports/sortSpecifiersText.ts","../src/rules/sortedImports/createFix/formatNamedImport.ts","../src/rules/sortedImports/createFix/buildSortedCode.ts","../src/rules/sortedImports/createFix/findLastImportIndex.ts","../src/rules/sortedImports/createFix/getReplacementRange.ts","../src/rules/sortedImports/createFix/groupImportsByType.ts","../src/rules/sortedImports/createFix/sortImportGroups.ts","../src/rules/sortedImports/createFix/index.ts","../src/rules/sortedImports/getImportDeclarations.ts","../src/rules/sortedImports/index.ts","../src/rules/sortedReExports/categorizeReExport.ts","../src/rules/sortedReExports/getSortKey.ts","../src/rules/sortedReExports/categorizeReExports.ts","../src/rules/sortedReExports/ReExportGroupOrder.ts","../src/rules/sortedReExports/checkAlphabeticalSorting.ts","../src/rules/sortedReExports/checkGroupOrdering.ts","../src/rules/sortedReExports/getSpecifierName.ts","../src/rules/sortedReExports/areSpecifiersSorted.ts","../src/rules/sortedReExports/getNamedSpecifiers.ts","../src/rules/sortedReExports/isNamedReExport.ts","../src/rules/sortedReExports/checkSpecifiersSorting.ts","../src/rules/sortedReExports/sortSpecifiersText.ts","../src/rules/sortedReExports/createFix/formatNamedReExport.ts","../src/rules/sortedReExports/createFix/buildSortedCode.ts","../src/rules/sortedReExports/createFix/findFirstExportIndex.ts","../src/rules/sortedReExports/createFix/findLastExportIndex.ts","../src/rules/sortedReExports/createFix/getReplacementRange.ts","../src/rules/sortedReExports/createFix/groupReExportsByType.ts","../src/rules/sortedReExports/createFix/sortExportGroups.ts","../src/rules/sortedReExports/createFix/index.ts","../src/rules/sortedReExports/getReExportDeclarations.ts","../src/rules/sortedReExports/index.ts"],"sourcesContent":["import eslint from '@eslint/js'\nimport react from 'eslint-plugin-react'\nimport reactHooks from 'eslint-plugin-react-hooks'\nimport stylistic from '@stylistic/eslint-plugin'\nimport typescript from 'typescript-eslint'\nimport {defineConfig} from 'eslint/config'\nimport {importsAndReExportsAtTop} from './rules/importsAndReExportsAtTop'\nimport {individualImports} from './rules/individualImports'\nimport {individualReExports} from './rules/individualReExports'\nimport {sortedImports} from './rules/sortedImports'\nimport {sortedReExports} from './rules/sortedReExports'\n\nexport const CONFIG = defineConfig([\n {\n ignores: [\n 'src/graphql/sdk.ts',\n '**/node_modules/**',\n '**/dist/**',\n ],\n },\n {\n settings: {\n react: {\n version: '19',\n },\n },\n },\n eslint.configs.recommended,\n react.configs.flat.recommended,\n stylistic.configs.recommended,\n typescript.configs.recommended,\n typescript.configs.stylistic,\n {\n plugins: {\n 'react-hooks': reactHooks,\n },\n rules: reactHooks.configs.recommended.rules,\n },\n {\n plugins: {\n '@borela-tech': {\n rules: {\n 'imports-and-re-exports-at-top': importsAndReExportsAtTop,\n 'individual-imports': individualImports,\n 'individual-re-exports': individualReExports,\n 'sorted-imports': sortedImports,\n 'sorted-re-exports': sortedReExports,\n },\n },\n },\n rules: {\n '@borela-tech/imports-and-re-exports-at-top': 'error',\n '@borela-tech/individual-imports': 'error',\n '@borela-tech/individual-re-exports': 'error',\n '@borela-tech/sorted-imports': 'error',\n '@borela-tech/sorted-re-exports': 'error',\n },\n },\n {\n rules: {\n 'capitalized-comments': [\n 'error',\n 'always',\n {ignoreConsecutiveComments: true},\n ],\n 'react/react-in-jsx-scope': 'off',\n '@stylistic/arrow-parens': [\n 'error',\n 'as-needed',\n ],\n '@stylistic/array-bracket-newline': [\n 'error',\n 'consistent',\n ],\n '@stylistic/array-bracket-spacing': [\n 'error',\n 'never',\n ],\n '@stylistic/array-element-newline': [\n 'error',\n 'consistent',\n ],\n '@stylistic/block-spacing': 'off',\n '@stylistic/brace-style': [\n 'error',\n '1tbs',\n {allowSingleLine: true},\n ],\n '@stylistic/indent': [\n 'error',\n 2,\n {ignoredNodes: ['TSMappedType > *']},\n ],\n '@stylistic/jsx-tag-spacing': [\n 'error',\n {\n afterOpening: 'never',\n beforeClosing: 'never',\n beforeSelfClosing: 'never',\n closingSlash: 'never',\n },\n ],\n '@stylistic/jsx-wrap-multilines': 'off',\n '@stylistic/lines-between-class-members': 'off',\n '@stylistic/object-curly-newline': [\n 'error',\n {consistent: true},\n ],\n '@stylistic/object-curly-spacing': [\n 'error',\n 'never',\n ],\n '@stylistic/operator-linebreak': [\n 'error',\n 'before',\n {overrides: {'=': 'after'}},\n ],\n '@stylistic/quotes': [\n 'error',\n 'single',\n {avoidEscape: true},\n ],\n '@stylistic/quote-props': [\n 'error',\n 'as-needed',\n ],\n '@stylistic/semi': [\n 'error',\n 'never',\n {beforeStatementContinuationChars: 'always'},\n ],\n '@typescript-eslint/no-empty-function': 'off',\n '@typescript-eslint/consistent-indexed-object-style': 'off',\n },\n },\n])\n","import {StatementType} from './statementType'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function getStatementType(statement: TSESTree.Statement): StatementType {\n if (statement.type === 'ImportDeclaration')\n return 'import'\n\n if (statement.type === 'ExportAllDeclaration')\n return 're-export'\n\n if (statement.type === 'ExportNamedDeclaration') {\n if (statement.source !== null)\n return 're-export'\n }\n\n return 'other'\n}\n","import {getStatementType} from './getStatementType'\nimport {ReExport} from './ReExport'\nimport type {CategorizedStatements} from './CategorizedStatements'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function categorizeStatements(\n statements: TSESTree.Statement[],\n): CategorizedStatements {\n const result: CategorizedStatements = {\n imports: [],\n reExports: [],\n other: [],\n }\n\n for (const statement of statements) {\n const type = getStatementType(statement)\n\n if (type === 'import')\n result.imports.push(statement as TSESTree.ImportDeclaration)\n else if (type === 're-export')\n result.reExports.push(statement as ReExport)\n else\n result.other.push(statement)\n }\n\n return result\n}\n","import {getStatementType} from './getStatementType'\nimport type {StatementIndices} from './StatementIndices'\nimport type {StatementType} from './statementType'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function findFirstIndices(\n statements: TSESTree.Statement[],\n): StatementIndices {\n let firstImport = Infinity\n let firstReExport = Infinity\n let firstOther = -1\n\n for (let i = 0; i < statements.length; i++) {\n const type: StatementType = getStatementType(statements[i])\n\n if (type === 'import' && firstImport === Infinity)\n firstImport = i\n else if (type === 're-export' && firstReExport === Infinity)\n firstReExport = i\n else if (type === 'other' && firstOther === -1)\n firstOther = i\n }\n\n return {firstImport, firstReExport, firstOther}\n}\n","import {Rule} from 'eslint'\nimport type {CategorizedStatements} from './CategorizedStatements'\nimport type {Node as ESTreeNode} from 'estree'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function generateSortedText(\n context: Rule.RuleContext,\n categories: CategorizedStatements,\n): string {\n const allStatements: TSESTree.Node[] = [\n ...categories.imports,\n ...categories.reExports,\n ...categories.other,\n ]\n return allStatements.map(\n node => context.sourceCode.getText(node as ESTreeNode),\n ).join('\\n')\n}\n","import type {CategorizedStatements} from './CategorizedStatements'\nimport type {StatementIndices} from './StatementIndices'\n\nexport function hasViolation(\n indices: StatementIndices,\n categories: CategorizedStatements,\n): boolean {\n const {\n firstImport,\n firstReExport,\n firstOther,\n } = indices\n\n // No imports or no re-exports.\n if (categories.imports.length === 0 || categories.reExports.length === 0)\n return false\n\n const firstImportOrReExport = Math.min(firstImport, firstReExport)\n const hasOtherBeforeImportOrReExport =\n firstOther !== -1 && firstOther < firstImportOrReExport\n\n // Violation if:\n // 1. Other statements appear before imports/re-exports.\n // 2. Re-exports appear before imports.\n if (hasOtherBeforeImportOrReExport || firstImport > firstReExport)\n return true\n\n return false\n}\n","import {categorizeStatements} from './categorizeStatements'\nimport {findFirstIndices} from './findFirstIndices'\nimport {generateSortedText} from './generateSortedText'\nimport {hasViolation} from './hasViolation'\nimport type {Rule} from 'eslint'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport const importsAndReExportsAtTop: Rule.RuleModule = {\n meta: {\n type: 'suggestion',\n docs: {\n description: 'Enforce imports and re-exports at the top of the file',\n recommended: false,\n },\n fixable: 'code',\n messages: {\n importsAndReExportsAtTop:\n 'Imports and re-exports should be at the top of the file.',\n },\n schema: [],\n },\n\n create(context) {\n return {\n Program(node) {\n const statements = node.body as TSESTree.Statement[]\n const categories = categorizeStatements(statements)\n const indices = findFirstIndices(statements)\n\n if (!hasViolation(indices, categories))\n return\n\n context.report({\n node,\n messageId: 'importsAndReExportsAtTop',\n\n fix(fixer) {\n const sortedText = generateSortedText(context, categories)\n return fixer.replaceText(node, sortedText)\n },\n })\n },\n }\n },\n}\n","import type {Rule} from 'eslint'\n\nexport const individualImports: Rule.RuleModule = {\n meta: {\n docs: {\n description: 'Enforce individual imports instead of grouped imports',\n recommended: true,\n },\n fixable: 'code',\n messages: {\n individualImports: 'Use individual imports instead of grouped imports.',\n },\n schema: [],\n type: 'suggestion',\n },\n create(context) {\n return {\n ImportDeclaration(node) {\n if (node.specifiers.length <= 1)\n return\n\n context.report({\n node,\n messageId: 'individualImports',\n fix(fixer) {\n const source = node.source.raw\n const specifiers = node.specifiers\n .filter(s => s.type === 'ImportSpecifier')\n .map(s => `import {${s.local.name}} from ${source}`)\n .join('\\n')\n return fixer.replaceText(node, specifiers)\n },\n })\n },\n }\n },\n}\n","import type {Rule} from 'eslint'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport const individualReExports: Rule.RuleModule = {\n meta: {\n docs: {\n description: 'Enforce individual exports instead of grouped exports',\n recommended: true,\n },\n fixable: 'code',\n messages: {\n individualReExports: 'Use individual exports instead of grouped exports.',\n },\n schema: [],\n type: 'suggestion',\n },\n create(context) {\n return {\n ExportNamedDeclaration(node) {\n const exportNode = node as TSESTree.ExportNamedDeclaration\n if (!exportNode.source || exportNode.specifiers.length <= 1)\n return\n\n context.report({\n node,\n messageId: 'individualReExports',\n fix(fixer) {\n const source = exportNode.source!.value\n const typeKeyword = exportNode.exportKind === 'type'\n ? 'type '\n : ''\n const specifiers = exportNode.specifiers\n .map(s => {\n const localName = s.local.type === 'Identifier'\n ? s.local.name\n : s.local.value\n const exportedName = s.exported.type === 'Identifier'\n ? s.exported.name\n : s.exported.value\n const name = localName === exportedName\n ? localName\n : `${localName} as ${exportedName}`\n return `export ${typeKeyword}{${name}} from '${source}'`\n })\n .join('\\n')\n return fixer.replaceText(node, specifiers)\n },\n })\n },\n }\n },\n}\n","import type {ImportGroup} from './ImportGroup'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function categorizeImport(declaration: TSESTree.ImportDeclaration): ImportGroup {\n // Example: import type {Type} from 'module'\n if (declaration.importKind === 'type')\n return 'type'\n\n // Example: import 'module'\n if (declaration.specifiers.length === 0)\n return 'side-effect'\n\n // Example: import value from 'module'\n if (declaration.specifiers.some(s => s.type === 'ImportDefaultSpecifier'))\n return 'default'\n\n // Example: import {value} from 'module'\n return 'named'\n}\n","import {categorizeImport} from './categorizeImport'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function getSortKey(declaration: TSESTree.ImportDeclaration): string {\n const group = categorizeImport(declaration)\n\n if (group === 'side-effect')\n return declaration.source.value\n\n if (group === 'default') {\n const defaultSpecifier = declaration.specifiers.find(\n s => s.type === 'ImportDefaultSpecifier',\n ) as TSESTree.ImportDefaultSpecifier | undefined\n\n return defaultSpecifier?.local.name ?? ''\n }\n\n const specifier = declaration.specifiers[0]\n return specifier.local.name\n}\n","import {categorizeImport} from './categorizeImport'\nimport {getSortKey} from './getSortKey'\nimport type {CategorizedImport} from './CategorizedImport'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function categorizeImports(declarations: TSESTree.ImportDeclaration[]): CategorizedImport[] {\n return declarations.map(declaration => ({\n declaration,\n group: categorizeImport(declaration),\n sortKey: getSortKey(declaration),\n }))\n}\n","export function compare(a: string, b: string): number {\n return a.localeCompare(b, 'en', {sensitivity: 'case'})\n}\n","import {ImportGroup} from './ImportGroup'\n\nexport const importGroupOrder: ImportGroup[] = [\n 'side-effect',\n 'default',\n 'named',\n 'type',\n]\n","import {compare} from '../../lib/compare'\nimport {importGroupOrder} from './ImportGroupOrder'\nimport type {CategorizedImport} from './CategorizedImport'\nimport type {ImportError} from './ImportError'\n\nexport function checkAlphabeticalSorting(categorized: CategorizedImport[]): ImportError[] {\n const errors: ImportError[] = []\n\n for (const group of importGroupOrder) {\n const groupImports = categorized.filter(c => c.group === group)\n const sorted = [...groupImports].sort((a, b) => compare(a.sortKey, b.sortKey))\n for (let i = 0; i < groupImports.length; i++) {\n if (groupImports[i] !== sorted[i]) {\n errors.push({\n node: groupImports[i].declaration,\n messageId: 'sortedImports',\n })\n }\n }\n }\n\n return errors\n}\n","import {importGroupOrder} from './ImportGroupOrder'\nimport type {CategorizedImport} from './CategorizedImport'\nimport type {ImportError} from './ImportError'\n\nexport function checkGroupOrdering(categorized: CategorizedImport[]): ImportError[] {\n const errors: ImportError[] = []\n\n let currentGroupIndex = -1\n for (const {declaration, group} of categorized) {\n const groupIndex = importGroupOrder.indexOf(group)\n if (groupIndex < currentGroupIndex) {\n errors.push({\n node: declaration,\n messageId: 'wrongGroup',\n })\n } else\n currentGroupIndex = groupIndex\n }\n\n return errors\n}\n","import type {TSESTree} from '@typescript-eslint/types'\n\nexport function getSpecifierName(specifier: TSESTree.ImportSpecifier): string {\n return specifier.imported.type === 'Identifier'\n ? specifier.imported.name\n : String(specifier.imported.value)\n}\n","import {compare} from '../../lib/compare'\nimport {getSpecifierName} from './getSpecifierName'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function areSpecifiersSorted(specifiers: TSESTree.ImportSpecifier[]): boolean {\n const names = specifiers.map(s => getSpecifierName(s))\n const sorted = [...names].sort((a, b) => compare(a, b))\n return names.every((name, i) => name === sorted[i])\n}\n","import type {TSESTree} from '@typescript-eslint/types'\n\nexport function getNamedSpecifiers(declaration: TSESTree.ImportDeclaration): TSESTree.ImportSpecifier[] {\n return declaration.specifiers.filter(\n (s): s is TSESTree.ImportSpecifier => s.type === 'ImportSpecifier',\n )\n}\n","import {areSpecifiersSorted} from './areSpecifiersSorted'\nimport {getNamedSpecifiers} from './getNamedSpecifiers'\nimport type {CategorizedImport} from './CategorizedImport'\nimport type {ImportError} from './ImportError'\n\nexport function checkSpecifiersSorting(categorized: CategorizedImport[]): ImportError[] {\n const errors: ImportError[] = []\n const namedImports = categorized.filter(c => c.group === 'named')\n\n for (const {declaration} of namedImports) {\n const specifiers = getNamedSpecifiers(declaration)\n if (specifiers.length > 1 && !areSpecifiersSorted(specifiers)) {\n errors.push({\n node: declaration,\n messageId: 'sortedNames',\n })\n }\n }\n\n return errors\n}\n","import {compare} from '../../lib/compare'\nimport {getSpecifierName} from './getSpecifierName'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function sortSpecifiersText(\n specifiers: TSESTree.ImportSpecifier[],\n sourceCode: {getText: (node: TSESTree.ImportSpecifier) => string},\n): string {\n const sorted = [...specifiers].sort((a, b) => {\n const nameA = getSpecifierName(a)\n const nameB = getSpecifierName(b)\n return compare(nameA, nameB)\n })\n return sorted.map(s => sourceCode.getText(s)).join(', ')\n}\n","import {areSpecifiersSorted} from '../areSpecifiersSorted'\nimport {getNamedSpecifiers} from '../getNamedSpecifiers'\nimport {sortSpecifiersText} from '../sortSpecifiersText'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function formatNamedImport(\n declaration: TSESTree.ImportDeclaration,\n sourceCode: {getText: (node?: unknown) => string},\n): string {\n const specifiers = getNamedSpecifiers(declaration)\n\n if (specifiers.length > 1 && !areSpecifiersSorted(specifiers)) {\n const sortedSpecifiers = sortSpecifiersText(specifiers, sourceCode)\n const source = declaration.source.value\n const prefix = declaration.importKind === 'type' ? 'import type ' : 'import '\n return `${prefix}{${sortedSpecifiers}} from '${source}'`\n }\n\n return sourceCode.getText(declaration)\n}\n","import {formatNamedImport} from './formatNamedImport'\nimport {importGroupOrder} from '../ImportGroupOrder'\nimport type {CategorizedImport} from '../CategorizedImport'\nimport type {ImportGroup} from '../ImportGroup'\n\nexport function buildSortedCode(\n grouped: Record<ImportGroup, CategorizedImport[]>,\n sourceCode: {getText: (node?: unknown) => string},\n): string[] {\n const sortedCode: string[] = []\n\n for (const group of importGroupOrder) {\n for (const {declaration} of grouped[group]) {\n if (group === 'named' || group === 'type')\n sortedCode.push(formatNamedImport(declaration, sourceCode))\n else\n sortedCode.push(sourceCode.getText(declaration))\n }\n }\n\n return sortedCode\n}\n","import type {TSESTree} from '@typescript-eslint/types'\n\nexport function findLastImportIndex(programBody: TSESTree.ProgramStatement[]): number {\n let lastIndex = 0\n for (let i = 0; i < programBody.length; i++) {\n if (programBody[i].type === 'ImportDeclaration')\n lastIndex = i\n else\n break\n }\n return lastIndex\n}\n","import {findLastImportIndex} from './findLastImportIndex'\nimport type {ReplacementRange} from '../../../lib/ReplacementRange'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function getReplacementRange(\n programBody: TSESTree.ProgramStatement[],\n): ReplacementRange {\n const lastIndex = findLastImportIndex(programBody)\n const firstImport = programBody[0] as TSESTree.ImportDeclaration\n const lastImport = programBody[lastIndex] as TSESTree.ImportDeclaration\n const start = firstImport.range![0]\n const end = lastImport.range![1]\n return {start, end}\n}\n","import type {CategorizedImport} from '../CategorizedImport'\nimport type {ImportGroup} from '../ImportGroup'\n\nexport function groupImportsByType(\n categorized: CategorizedImport[],\n): Record<ImportGroup, CategorizedImport[]> {\n const grouped: Record<ImportGroup, CategorizedImport[]> = {\n 'side-effect': [],\n default: [],\n named: [],\n type: [],\n }\n\n for (const item of categorized)\n grouped[item.group].push(item)\n\n return grouped\n}\n","import {compare} from '../../../lib/compare'\nimport type {CategorizedImport} from '../CategorizedImport'\nimport type {ImportGroup} from '../ImportGroup'\n\nexport function sortImportGroups(\n grouped: Record<ImportGroup, CategorizedImport[]>,\n): void {\n grouped['side-effect'].sort((a, b) => compare(a.sortKey, b.sortKey))\n grouped['default'].sort((a, b) => compare(a.sortKey, b.sortKey))\n grouped['named'].sort((a, b) => compare(a.sortKey, b.sortKey))\n grouped['type'].sort((a, b) => compare(a.sortKey, b.sortKey))\n}\n","import {buildSortedCode} from './buildSortedCode'\nimport {categorizeImports} from '../categorizeImports'\nimport {getReplacementRange} from './getReplacementRange'\nimport {groupImportsByType} from './groupImportsByType'\nimport {sortImportGroups} from './sortImportGroups'\nimport type {Rule} from 'eslint'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function createFix(\n fixer: Rule.RuleFixer,\n importDeclarations: TSESTree.ImportDeclaration[],\n sourceCode: {getText: (node?: unknown) => string},\n programBody: TSESTree.ProgramStatement[],\n) {\n const range = getReplacementRange(programBody)\n const categorized = categorizeImports(importDeclarations)\n const grouped = groupImportsByType(categorized)\n\n sortImportGroups(grouped)\n\n const sortedCode = buildSortedCode(grouped, sourceCode)\n .join('\\n')\n\n return fixer.replaceTextRange(\n [range.start, range.end],\n sortedCode,\n )\n}\n","import type {TSESTree} from '@typescript-eslint/types'\n\nexport function getImportDeclarations(programBody: TSESTree.ProgramStatement[]): TSESTree.ImportDeclaration[] {\n return programBody.filter(\n (statement): statement is TSESTree.ImportDeclaration =>\n statement.type === 'ImportDeclaration',\n )\n}\n","import {categorizeImports} from './categorizeImports'\nimport {checkAlphabeticalSorting} from './checkAlphabeticalSorting'\nimport {checkGroupOrdering} from './checkGroupOrdering'\nimport {checkSpecifiersSorting} from './checkSpecifiersSorting'\nimport {createFix} from './createFix'\nimport {getImportDeclarations} from './getImportDeclarations'\nimport type {ImportError} from './ImportError'\nimport type {Rule} from 'eslint'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport const sortedImports: Rule.RuleModule = {\n meta: {\n docs: {\n description: 'Enforce sorted imports alphabetically',\n recommended: true,\n },\n fixable: 'code',\n messages: {\n sortedImports: 'Imports should be sorted alphabetically',\n sortedNames: 'Named imports should be sorted alphabetically',\n wrongGroup: 'Import is in wrong group',\n },\n schema: [],\n type: 'suggestion',\n },\n create(context) {\n return {\n Program(node) {\n const body = node.body as TSESTree.ProgramStatement[]\n const declarations = getImportDeclarations(body)\n if (declarations.length === 0)\n return\n\n const categorized = categorizeImports(declarations)\n const errors: ImportError[] = [\n ...checkGroupOrdering(categorized),\n ...checkAlphabeticalSorting(categorized),\n ...checkSpecifiersSorting(categorized),\n ]\n\n for (const error of errors) {\n context.report({\n node: error.node,\n messageId: error.messageId,\n fix(fixer) {\n const sourceCode = context.sourceCode\n return createFix(fixer, declarations, sourceCode, body)\n },\n })\n }\n },\n }\n },\n}\n","import type {ReExportGroup} from './ReExportGroup'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function categorizeReExport(\n declaration: TSESTree.ExportNamedDeclaration | TSESTree.ExportAllDeclaration,\n): ReExportGroup {\n // Example: export * from 'module'\n if (declaration.type === 'ExportAllDeclaration')\n return 're-export-all'\n\n // Example: export type {Type} from 'module'\n if (declaration.exportKind === 'type')\n return 're-export-type'\n\n // Example: export {value} from 'module'\n return 're-export-named'\n}\n","import type {TSESTree} from '@typescript-eslint/types'\n\nexport function getSortKey(\n declaration: TSESTree.ExportNamedDeclaration | TSESTree.ExportAllDeclaration,\n): string {\n if (declaration.type === 'ExportAllDeclaration')\n return declaration.source.value\n\n const specifier = declaration.specifiers[0]\n if (!specifier)\n return ''\n\n return specifier.local.type === 'Identifier'\n ? specifier.local.name\n : specifier.local.value\n}\n","import {categorizeReExport} from './categorizeReExport'\nimport {getSortKey} from './getSortKey'\nimport {ReExportDeclaration} from './ReExportDeclaration'\nimport type {CategorizedReExport} from './CategorizedReExport'\n\nexport function categorizeReExports(declarations: ReExportDeclaration[]): CategorizedReExport[] {\n return declarations.map(declaration => {\n return {\n declaration,\n group: categorizeReExport(declaration),\n sortKey: getSortKey(declaration),\n } as CategorizedReExport\n })\n}\n","import {ReExportGroup} from './ReExportGroup'\n\nexport const reExportGroupOrder: ReExportGroup[] = [\n 're-export-all',\n 're-export-named',\n 're-export-type',\n]\n","import {compare} from '../../lib/compare'\nimport {reExportGroupOrder} from './ReExportGroupOrder'\nimport type {CategorizedReExport} from './CategorizedReExport'\nimport type {ReExportError} from './ReExportError'\n\nexport function checkAlphabeticalSorting(categorized: CategorizedReExport[]): ReExportError[] {\n const errors: ReExportError[] = []\n\n for (const group of reExportGroupOrder) {\n const groupReExports = categorized.filter(c => c.group === group)\n const sorted = [...groupReExports].sort((a, b) =>\n compare(a.sortKey, b.sortKey),\n )\n for (let i = 0; i < groupReExports.length; i++) {\n if (groupReExports[i] !== sorted[i]) {\n errors.push({\n node: groupReExports[i].declaration,\n messageId: 'sortedReExports',\n })\n }\n }\n }\n\n return errors\n}\n","import {reExportGroupOrder} from './ReExportGroupOrder'\nimport type {CategorizedReExport} from './CategorizedReExport'\nimport type {ReExportError} from './ReExportError'\n\nexport function checkGroupOrdering(categorized: CategorizedReExport[]): ReExportError[] {\n const errors: ReExportError[] = []\n\n let currentGroupIndex = -1\n for (const {declaration, group} of categorized) {\n const groupIndex = reExportGroupOrder.indexOf(group)\n if (groupIndex < currentGroupIndex) {\n errors.push({\n node: declaration,\n messageId: 'wrongGroup',\n })\n } else\n currentGroupIndex = groupIndex\n }\n\n return errors\n}\n","import type {TSESTree} from '@typescript-eslint/types'\n\nexport function getSpecifierName(specifier: TSESTree.ExportSpecifier): string {\n return specifier.local.type === 'Identifier'\n ? specifier.local.name\n : String(specifier.local.value)\n}\n","import {compare} from '../../lib/compare'\nimport {getSpecifierName} from './getSpecifierName'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function areSpecifiersSorted(specifiers: TSESTree.ExportSpecifier[]): boolean {\n const names = specifiers.map(s => getSpecifierName(s))\n const sorted = [...names].sort((a, b) => compare(a, b))\n return names.every((name, i) => name === sorted[i])\n}\n","import type {TSESTree} from '@typescript-eslint/types'\n\nexport function getNamedSpecifiers(\n declaration: TSESTree.ExportNamedDeclaration,\n): TSESTree.ExportSpecifier[] {\n return declaration.specifiers.filter(\n s => s.type === 'ExportSpecifier' && s.local.type === 'Identifier',\n ) as TSESTree.ExportSpecifier[]\n}\n","import {CategorizedNamedReExport} from './CategorizedNamedReExport'\nimport type {CategorizedReExport} from './CategorizedReExport'\n\nexport function isNamedReExport(x: CategorizedReExport): x is CategorizedNamedReExport {\n return x.group !== 're-export-all'\n}\n","import {areSpecifiersSorted} from './areSpecifiersSorted'\nimport {getNamedSpecifiers} from './getNamedSpecifiers'\nimport {isNamedReExport} from './isNamedReExport'\nimport type {CategorizedReExport} from './CategorizedReExport'\nimport type {ReExportError} from './ReExportError'\n\nexport function checkSpecifiersSorting(categorized: CategorizedReExport[]): ReExportError[] {\n const errors: ReExportError[] = []\n const namedReExports = categorized.filter(isNamedReExport)\n\n for (const {declaration} of namedReExports) {\n const specifiers = getNamedSpecifiers(declaration)\n const isSorted = areSpecifiersSorted(specifiers)\n if (specifiers.length > 1 && !isSorted) {\n errors.push({\n node: declaration,\n messageId: 'sortedNames',\n })\n }\n }\n\n return errors\n}\n","import {compare} from '../../lib/compare'\nimport {getSpecifierName} from './getSpecifierName'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function sortSpecifiersText(\n specifiers: TSESTree.ExportSpecifier[],\n sourceCode: {getText: (node?: unknown) => string},\n): string {\n const sorted = [...specifiers].sort((a, b) => {\n const nameA = getSpecifierName(a)\n const nameB = getSpecifierName(b)\n return compare(nameA, nameB)\n })\n return sorted.map(s => sourceCode.getText(s)).join(', ')\n}\n","import {areSpecifiersSorted} from '../areSpecifiersSorted'\nimport {getNamedSpecifiers} from '../getNamedSpecifiers'\nimport {sortSpecifiersText} from '../sortSpecifiersText'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function formatNamedReExport(\n declaration: TSESTree.ExportNamedDeclaration,\n sourceCode: {getText: (node?: unknown) => string},\n): string {\n const specifiers = getNamedSpecifiers(declaration)\n\n if (specifiers.length > 1 && !areSpecifiersSorted(specifiers)) {\n const sortedSpecifiers = sortSpecifiersText(specifiers, sourceCode)\n const source = declaration.source!.value\n const prefix = declaration.exportKind === 'type' ? 'export type ' : 'export '\n return `${prefix}{${sortedSpecifiers}} from '${source}'`\n }\n\n return sourceCode.getText(declaration)\n}\n","import {formatNamedReExport} from './formatNamedReExport'\nimport {isNamedReExport} from '../isNamedReExport'\nimport {reExportGroupOrder} from '../ReExportGroupOrder'\nimport type {CategorizedReExport} from '../CategorizedReExport'\nimport type {ReExportGroup} from '../ReExportGroup'\n\nexport function buildSortedCode(\n grouped: Record<ReExportGroup, CategorizedReExport[]>,\n sourceCode: {getText: (node?: unknown) => string},\n): string[] {\n const sortedCode: string[] = []\n\n for (const group of reExportGroupOrder) {\n for (const item of grouped[group]) {\n if (isNamedReExport(item)) {\n sortedCode.push(\n formatNamedReExport(\n item.declaration,\n sourceCode,\n ),\n )\n } else\n sortedCode.push(sourceCode.getText(item.declaration))\n }\n }\n\n return sortedCode\n}\n","import type {TSESTree} from '@typescript-eslint/types'\n\nexport function findFirstExportIndex(programBody: TSESTree.ProgramStatement[]): number {\n for (let i = 0; i < programBody.length; i++) {\n if (programBody[i].type === 'ExportNamedDeclaration'\n || programBody[i].type === 'ExportAllDeclaration') {\n return i\n }\n }\n return -1\n}\n","import type {TSESTree} from '@typescript-eslint/types'\n\nexport function findLastExportIndex(programBody: TSESTree.ProgramStatement[]): number {\n let lastIndex = -1\n for (let i = 0; i < programBody.length; i++) {\n if (programBody[i].type === 'ExportNamedDeclaration'\n || programBody[i].type === 'ExportAllDeclaration') {\n lastIndex = i\n }\n }\n return lastIndex\n}\n","import {findFirstExportIndex} from './findFirstExportIndex'\nimport {findLastExportIndex} from './findLastExportIndex'\nimport type {ReplacementRange} from '../../../lib/ReplacementRange'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function getReplacementRange(\n programBody: TSESTree.ProgramStatement[],\n): ReplacementRange {\n const firstIndex = findFirstExportIndex(programBody)\n const lastIndex = findLastExportIndex(programBody)\n\n if (firstIndex === -1 || lastIndex === -1)\n return {start: 0, end: 0}\n\n const firstExport = programBody[firstIndex]\n const lastExport = programBody[lastIndex]\n\n const start = firstExport.range[0]\n const end = lastExport.range[1]\n\n return {start, end}\n}\n","import type {CategorizedReExport} from '../CategorizedReExport'\nimport type {ReExportGroup} from '../ReExportGroup'\n\nexport function groupReExportsByType(\n categorized: CategorizedReExport[],\n): Record<ReExportGroup, CategorizedReExport[]> {\n const grouped: Record<ReExportGroup, CategorizedReExport[]> = {\n 're-export-all': [],\n 're-export-named': [],\n 're-export-type': [],\n }\n\n for (const item of categorized)\n grouped[item.group].push(item)\n\n return grouped\n}\n","import {compare} from '../../../lib/compare'\nimport type {CategorizedReExport} from '../CategorizedReExport'\nimport type {ReExportGroup} from '../ReExportGroup'\n\nexport function sortExportGroups(\n grouped: Record<ReExportGroup, CategorizedReExport[]>,\n): void {\n grouped['re-export-all'].sort((a, b) => compare(a.sortKey, b.sortKey))\n grouped['re-export-named'].sort((a, b) => compare(a.sortKey, b.sortKey))\n grouped['re-export-type'].sort((a, b) => compare(a.sortKey, b.sortKey))\n}\n","import {buildSortedCode} from './buildSortedCode'\nimport {categorizeReExports} from '../categorizeReExports'\nimport {getReplacementRange} from './getReplacementRange'\nimport {groupReExportsByType} from './groupReExportsByType'\nimport {ReExportDeclaration} from '../ReExportDeclaration'\nimport {sortExportGroups} from './sortExportGroups'\nimport type {Rule} from 'eslint'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function createFix(\n fixer: Rule.RuleFixer,\n reExportDeclarations: ReExportDeclaration[],\n sourceCode: {getText: (node?: unknown) => string},\n programBody: TSESTree.ProgramStatement[],\n) {\n const range = getReplacementRange(programBody)\n const categorized = categorizeReExports(reExportDeclarations)\n const grouped = groupReExportsByType(categorized)\n\n sortExportGroups(grouped)\n\n const sortedCode = buildSortedCode(grouped, sourceCode)\n .join('\\n')\n\n return fixer.replaceTextRange(\n [range.start, range.end],\n sortedCode,\n )\n}\n","import type {ReExportDeclaration} from './ReExportDeclaration'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function getReExportDeclarations(\n programBody: TSESTree.ProgramStatement[],\n): ReExportDeclaration[] {\n return programBody.filter(\n (statement): statement is ReExportDeclaration =>\n (statement.type === 'ExportNamedDeclaration' && statement.source !== null)\n || statement.type === 'ExportAllDeclaration',\n )\n}\n","import {categorizeReExports} from './categorizeReExports'\nimport {checkAlphabeticalSorting} from './checkAlphabeticalSorting'\nimport {checkGroupOrdering} from './checkGroupOrdering'\nimport {checkSpecifiersSorting} from './checkSpecifiersSorting'\nimport {createFix} from './createFix'\nimport {getReExportDeclarations} from './getReExportDeclarations'\nimport type {ReExportError} from './ReExportError'\nimport type {Rule} from 'eslint'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport const sortedReExports: Rule.RuleModule = {\n meta: {\n docs: {\n description: 'Enforce sorted exports alphabetically',\n recommended: true,\n },\n fixable: 'code',\n messages: {\n sortedReExports: 'Exports should be sorted alphabetically',\n sortedNames: 'Named exports should be sorted alphabetically',\n wrongGroup: 'Export is in wrong group',\n },\n schema: [],\n type: 'suggestion',\n },\n create(context) {\n return {\n Program(node) {\n const body = node.body as TSESTree.ProgramStatement[]\n const declarations = getReExportDeclarations(body)\n if (declarations.length === 0)\n return\n\n const categorized = categorizeReExports(declarations)\n const errors: ReExportError[] = [\n ...checkGroupOrdering(categorized),\n ...checkAlphabeticalSorting(categorized),\n ...checkSpecifiersSorting(categorized),\n ]\n\n for (const error of errors) {\n context.report({\n node: error.node,\n messageId: error.messageId,\n fix(fixer) {\n const sourceCode = context.sourceCode\n return createFix(fixer, declarations, sourceCode, body)\n },\n })\n }\n },\n }\n },\n}\n"],"mappings":"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","names":["getSortKey","getSortKey","checkAlphabeticalSorting","checkGroupOrdering","getSpecifierName","areSpecifiersSorted","getSpecifierName","getNamedSpecifiers","checkSpecifiersSorting","getNamedSpecifiers","areSpecifiersSorted","sortSpecifiersText","getSpecifierName","getNamedSpecifiers","areSpecifiersSorted","sortSpecifiersText","buildSortedCode","getReplacementRange","createFix","getReplacementRange","buildSortedCode","checkGroupOrdering","checkAlphabeticalSorting","checkSpecifiersSorting","createFix"]}
1
+ {"version":3,"sources":["../src/index.ts","../src/rules/importsAndReExportsAtTop/getStatementType.ts","../src/rules/importsAndReExportsAtTop/categorizeStatements.ts","../src/rules/importsAndReExportsAtTop/findStatementIndices.ts","../src/rules/importsAndReExportsAtTop/generateSortedText.ts","../src/rules/importsAndReExportsAtTop/hasViolation.ts","../src/rules/importsAndReExportsAtTop/index.ts","../src/rules/individualImports.ts","../src/rules/individualReExports.ts","../src/rules/sortedImports/categorizeImport.ts","../src/rules/sortedImports/getSortKey.ts","../src/rules/sortedImports/categorizeImports.ts","../src/lib/compare.ts","../src/rules/sortedImports/ImportGroupOrder.ts","../src/rules/sortedImports/checkAlphabeticalSorting.ts","../src/rules/sortedImports/checkGroupOrdering.ts","../src/rules/sortedImports/getSpecifierName.ts","../src/rules/sortedImports/areSpecifiersSorted.ts","../src/rules/sortedImports/getNamedSpecifiers.ts","../src/rules/sortedImports/checkSpecifiersSorting.ts","../src/rules/sortedImports/sortSpecifiersText.ts","../src/rules/sortedImports/createFix/formatNamedImport.ts","../src/rules/sortedImports/createFix/buildSortedCode.ts","../src/rules/sortedImports/createFix/groupImportsByType.ts","../src/rules/sortedImports/createFix/sortImportGroups.ts","../src/rules/sortedImports/createFix/index.ts","../src/rules/sortedImports/getImportGroups.ts","../src/rules/sortedImports/index.ts","../src/rules/sortedReExports/categorizeReExport.ts","../src/rules/sortedReExports/getSortKey.ts","../src/rules/sortedReExports/categorizeReExports.ts","../src/rules/sortedReExports/ReExportGroupOrder.ts","../src/rules/sortedReExports/checkAlphabeticalSorting.ts","../src/rules/sortedReExports/checkGroupOrdering.ts","../src/rules/sortedReExports/getSpecifierName.ts","../src/rules/sortedReExports/areSpecifiersSorted.ts","../src/rules/sortedReExports/getNamedSpecifiers.ts","../src/rules/sortedReExports/isNamedReExport.ts","../src/rules/sortedReExports/checkSpecifiersSorting.ts","../src/rules/sortedReExports/sortSpecifiersText.ts","../src/rules/sortedReExports/createFix/formatNamedReExport.ts","../src/rules/sortedReExports/createFix/buildSortedCode.ts","../src/rules/sortedReExports/createFix/groupReExportsByType.ts","../src/rules/sortedReExports/createFix/sortExportGroups.ts","../src/rules/sortedReExports/createFix/index.ts","../src/rules/sortedReExports/getReExportGroups.ts","../src/rules/sortedReExports/index.ts"],"sourcesContent":["import eslint from '@eslint/js'\nimport react from 'eslint-plugin-react'\nimport reactHooks from 'eslint-plugin-react-hooks'\nimport stylistic from '@stylistic/eslint-plugin'\nimport typescript from 'typescript-eslint'\nimport {defineConfig} from 'eslint/config'\nimport {importsAndReExportsAtTop} from './rules/importsAndReExportsAtTop'\nimport {individualImports} from './rules/individualImports'\nimport {individualReExports} from './rules/individualReExports'\nimport {sortedImports} from './rules/sortedImports'\nimport {sortedReExports} from './rules/sortedReExports'\n\nexport const CONFIG = defineConfig([\n {\n ignores: [\n 'src/graphql/sdk.ts',\n '**/node_modules/**',\n '**/dist/**',\n ],\n },\n {\n settings: {\n react: {\n version: '19',\n },\n },\n },\n eslint.configs.recommended,\n react.configs.flat.recommended,\n stylistic.configs.recommended,\n typescript.configs.recommended,\n typescript.configs.stylistic,\n {\n plugins: {\n 'react-hooks': reactHooks,\n },\n rules: reactHooks.configs.recommended.rules,\n },\n {\n plugins: {\n '@borela-tech': {\n rules: {\n 'imports-and-re-exports-at-top': importsAndReExportsAtTop,\n 'individual-imports': individualImports,\n 'individual-re-exports': individualReExports,\n 'sorted-imports': sortedImports,\n 'sorted-re-exports': sortedReExports,\n },\n },\n },\n rules: {\n '@borela-tech/imports-and-re-exports-at-top': 'error',\n '@borela-tech/individual-imports': 'error',\n '@borela-tech/individual-re-exports': 'error',\n '@borela-tech/sorted-imports': 'error',\n '@borela-tech/sorted-re-exports': 'error',\n },\n },\n {\n rules: {\n 'capitalized-comments': [\n 'error',\n 'always',\n {ignoreConsecutiveComments: true},\n ],\n 'react/react-in-jsx-scope': 'off',\n '@stylistic/arrow-parens': [\n 'error',\n 'as-needed',\n ],\n '@stylistic/array-bracket-newline': [\n 'error',\n 'consistent',\n ],\n '@stylistic/array-bracket-spacing': [\n 'error',\n 'never',\n ],\n '@stylistic/array-element-newline': [\n 'error',\n 'consistent',\n ],\n '@stylistic/block-spacing': 'off',\n '@stylistic/brace-style': [\n 'error',\n '1tbs',\n {allowSingleLine: true},\n ],\n '@stylistic/indent': [\n 'error',\n 2,\n {ignoredNodes: ['TSMappedType > *']},\n ],\n '@stylistic/jsx-tag-spacing': [\n 'error',\n {\n afterOpening: 'never',\n beforeClosing: 'never',\n beforeSelfClosing: 'never',\n closingSlash: 'never',\n },\n ],\n '@stylistic/jsx-wrap-multilines': 'off',\n '@stylistic/lines-between-class-members': 'off',\n '@stylistic/object-curly-newline': [\n 'error',\n {consistent: true},\n ],\n '@stylistic/object-curly-spacing': [\n 'error',\n 'never',\n ],\n '@stylistic/operator-linebreak': [\n 'error',\n 'before',\n {overrides: {'=': 'after'}},\n ],\n '@stylistic/quotes': [\n 'error',\n 'single',\n {avoidEscape: true},\n ],\n '@stylistic/quote-props': [\n 'error',\n 'as-needed',\n ],\n '@stylistic/semi': [\n 'error',\n 'never',\n {beforeStatementContinuationChars: 'always'},\n ],\n '@typescript-eslint/no-empty-function': 'off',\n '@typescript-eslint/consistent-indexed-object-style': 'off',\n },\n },\n])\n","import {StatementType} from './statementType'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function getStatementType(statement: TSESTree.Statement): StatementType {\n if (statement.type === 'ImportDeclaration')\n return 'import'\n\n if (statement.type === 'ExportAllDeclaration')\n return 're-export'\n\n if (statement.type === 'ExportNamedDeclaration') {\n if (statement.source !== null)\n return 're-export'\n }\n\n return 'other'\n}\n","import {getStatementType} from './getStatementType'\nimport {ReExport} from './ReExport'\nimport type {CategorizedStatements} from './CategorizedStatements'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function categorizeStatements(\n statements: TSESTree.Statement[],\n): CategorizedStatements {\n const result: CategorizedStatements = {\n imports: [],\n reExports: [],\n other: [],\n }\n\n for (const statement of statements) {\n const type = getStatementType(statement)\n\n if (type === 'import')\n result.imports.push(statement as TSESTree.ImportDeclaration)\n else if (type === 're-export')\n result.reExports.push(statement as ReExport)\n else\n result.other.push(statement)\n }\n\n return result\n}\n","import {getStatementType} from './getStatementType'\nimport type {StatementIndices} from './StatementIndices'\nimport type {StatementType} from './statementType'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function findStatementIndices(\n statements: TSESTree.Statement[],\n): StatementIndices {\n let firstRegularStatement = -1\n let lastImport = -1\n let lastReExport = -1\n\n for (let i = 0; i < statements.length; i++) {\n const type: StatementType = getStatementType(statements[i])\n\n if (type === 'import')\n lastImport = i\n else if (type === 're-export')\n lastReExport = i\n else if (type === 'other' && firstRegularStatement === -1)\n firstRegularStatement = i\n }\n\n return {firstRegularStatement, lastImport, lastReExport}\n}\n","import {Rule} from 'eslint'\nimport type {CategorizedStatements} from './CategorizedStatements'\nimport type {Node as ESTreeNode} from 'estree'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function generateSortedText(\n context: Rule.RuleContext,\n categories: CategorizedStatements,\n): string {\n const allStatements: TSESTree.Node[] = [\n ...categories.imports,\n ...categories.reExports,\n ...categories.other,\n ]\n return allStatements.map(\n node => context.sourceCode.getText(node as ESTreeNode),\n ).join('\\n')\n}\n","import type {CategorizedStatements} from './CategorizedStatements'\nimport type {StatementIndices} from './StatementIndices'\n\nexport function hasViolation(\n indices: StatementIndices,\n categories: CategorizedStatements,\n): boolean {\n const {firstRegularStatement, lastImport, lastReExport} = indices\n\n if (categories.imports.length === 0 && categories.reExports.length === 0)\n return false\n\n const hasImportAfterRegularStatement = (\n categories.imports.length > 0\n && firstRegularStatement !== -1\n && lastImport > firstRegularStatement\n )\n const hasReExportAfterRegularStatement = (\n categories.reExports.length > 0\n && firstRegularStatement !== -1\n && lastReExport > firstRegularStatement\n )\n\n return hasImportAfterRegularStatement || hasReExportAfterRegularStatement\n}\n","import {categorizeStatements} from './categorizeStatements'\nimport {findStatementIndices} from './findStatementIndices'\nimport {generateSortedText} from './generateSortedText'\nimport {hasViolation} from './hasViolation'\nimport type {Rule} from 'eslint'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport const importsAndReExportsAtTop: Rule.RuleModule = {\n meta: {\n type: 'suggestion',\n docs: {\n description: 'Enforce imports and re-exports at the top of the file',\n recommended: false,\n },\n fixable: 'code',\n messages: {\n importsAndReExportsAtTop:\n 'Imports and re-exports should be at the top of the file.',\n },\n schema: [],\n },\n\n create(context) {\n return {\n Program(node) {\n const statements = node.body as TSESTree.Statement[]\n const categories = categorizeStatements(statements)\n const indices = findStatementIndices(statements)\n\n if (!hasViolation(indices, categories))\n return\n\n context.report({\n node,\n messageId: 'importsAndReExportsAtTop',\n\n fix(fixer) {\n const sortedText = generateSortedText(context, categories)\n return fixer.replaceText(node, sortedText)\n },\n })\n },\n }\n },\n}\n","import type {Rule} from 'eslint'\n\nexport const individualImports: Rule.RuleModule = {\n meta: {\n docs: {\n description: 'Enforce individual imports instead of grouped imports',\n recommended: true,\n },\n fixable: 'code',\n messages: {\n individualImports: 'Use individual imports instead of grouped imports.',\n },\n schema: [],\n type: 'suggestion',\n },\n create(context) {\n return {\n ImportDeclaration(node) {\n if (node.specifiers.length <= 1)\n return\n\n context.report({\n node,\n messageId: 'individualImports',\n fix(fixer) {\n const source = node.source.raw\n const specifiers = node.specifiers\n .filter(s => s.type === 'ImportSpecifier')\n .map(s => `import {${s.local.name}} from ${source}`)\n .join('\\n')\n return fixer.replaceText(node, specifiers)\n },\n })\n },\n }\n },\n}\n","import type {Rule} from 'eslint'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport const individualReExports: Rule.RuleModule = {\n meta: {\n docs: {\n description: 'Enforce individual exports instead of grouped exports',\n recommended: true,\n },\n fixable: 'code',\n messages: {\n individualReExports: 'Use individual exports instead of grouped exports.',\n },\n schema: [],\n type: 'suggestion',\n },\n create(context) {\n return {\n ExportNamedDeclaration(node) {\n const exportNode = node as TSESTree.ExportNamedDeclaration\n if (!exportNode.source || exportNode.specifiers.length <= 1)\n return\n\n context.report({\n node,\n messageId: 'individualReExports',\n fix(fixer) {\n const source = exportNode.source!.value\n const typeKeyword = exportNode.exportKind === 'type'\n ? 'type '\n : ''\n const specifiers = exportNode.specifiers\n .map(s => {\n const localName = s.local.type === 'Identifier'\n ? s.local.name\n : s.local.value\n const exportedName = s.exported.type === 'Identifier'\n ? s.exported.name\n : s.exported.value\n const name = localName === exportedName\n ? localName\n : `${localName} as ${exportedName}`\n return `export ${typeKeyword}{${name}} from '${source}'`\n })\n .join('\\n')\n return fixer.replaceText(node, specifiers)\n },\n })\n },\n }\n },\n}\n","import type {ImportGroup} from './ImportGroup'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function categorizeImport(declaration: TSESTree.ImportDeclaration): ImportGroup {\n // Example: import type {Type} from 'module'\n if (declaration.importKind === 'type')\n return 'type'\n\n // Example: import 'module'\n if (declaration.specifiers.length === 0)\n return 'side-effect'\n\n // Example: import value from 'module'\n if (declaration.specifiers.some(s => s.type === 'ImportDefaultSpecifier'))\n return 'default'\n\n // Example: import {value} from 'module'\n return 'named'\n}\n","import {categorizeImport} from './categorizeImport'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function getSortKey(declaration: TSESTree.ImportDeclaration): string {\n const group = categorizeImport(declaration)\n\n if (group === 'side-effect')\n return declaration.source.value\n\n if (group === 'default') {\n const defaultSpecifier = declaration.specifiers.find(\n s => s.type === 'ImportDefaultSpecifier',\n ) as TSESTree.ImportDefaultSpecifier | undefined\n\n return defaultSpecifier?.local.name ?? ''\n }\n\n const specifier = declaration.specifiers[0]\n return specifier.local.name\n}\n","import {categorizeImport} from './categorizeImport'\nimport {getSortKey} from './getSortKey'\nimport type {CategorizedImport} from './CategorizedImport'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function categorizeImports(declarations: TSESTree.ImportDeclaration[]): CategorizedImport[] {\n return declarations.map(declaration => ({\n declaration,\n group: categorizeImport(declaration),\n sortKey: getSortKey(declaration),\n }))\n}\n","export function compare(a: string, b: string): number {\n return a.localeCompare(b, 'en', {sensitivity: 'case'})\n}\n","import {ImportGroup} from './ImportGroup'\n\nexport const importGroupOrder: ImportGroup[] = [\n 'side-effect',\n 'default',\n 'named',\n 'type',\n]\n","import {compare} from '../../lib/compare'\nimport {importGroupOrder} from './ImportGroupOrder'\nimport type {CategorizedImport} from './CategorizedImport'\nimport type {ImportError} from './ImportError'\n\nexport function checkAlphabeticalSorting(categorized: CategorizedImport[]): ImportError[] {\n const errors: ImportError[] = []\n\n for (const group of importGroupOrder) {\n const groupImports = categorized.filter(c => c.group === group)\n const sorted = [...groupImports].sort((a, b) => compare(a.sortKey, b.sortKey))\n for (let i = 0; i < groupImports.length; i++) {\n if (groupImports[i] !== sorted[i]) {\n errors.push({\n node: groupImports[i].declaration,\n messageId: 'sortedImports',\n })\n }\n }\n }\n\n return errors\n}\n","import {importGroupOrder} from './ImportGroupOrder'\nimport type {CategorizedImport} from './CategorizedImport'\nimport type {ImportError} from './ImportError'\n\nexport function checkGroupOrdering(categorized: CategorizedImport[]): ImportError[] {\n const errors: ImportError[] = []\n\n let currentGroupIndex = -1\n for (const {declaration, group} of categorized) {\n const groupIndex = importGroupOrder.indexOf(group)\n if (groupIndex < currentGroupIndex) {\n errors.push({\n node: declaration,\n messageId: 'wrongGroup',\n })\n } else\n currentGroupIndex = groupIndex\n }\n\n return errors\n}\n","import type {TSESTree} from '@typescript-eslint/types'\n\nexport function getSpecifierName(specifier: TSESTree.ImportSpecifier): string {\n return specifier.imported.type === 'Identifier'\n ? specifier.imported.name\n : String(specifier.imported.value)\n}\n","import {compare} from '../../lib/compare'\nimport {getSpecifierName} from './getSpecifierName'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function areSpecifiersSorted(specifiers: TSESTree.ImportSpecifier[]): boolean {\n const names = specifiers.map(s => getSpecifierName(s))\n const sorted = [...names].sort((a, b) => compare(a, b))\n return names.every((name, i) => name === sorted[i])\n}\n","import type {TSESTree} from '@typescript-eslint/types'\n\nexport function getNamedSpecifiers(declaration: TSESTree.ImportDeclaration): TSESTree.ImportSpecifier[] {\n return declaration.specifiers.filter(\n (s): s is TSESTree.ImportSpecifier => s.type === 'ImportSpecifier',\n )\n}\n","import {areSpecifiersSorted} from './areSpecifiersSorted'\nimport {getNamedSpecifiers} from './getNamedSpecifiers'\nimport type {CategorizedImport} from './CategorizedImport'\nimport type {ImportError} from './ImportError'\n\nexport function checkSpecifiersSorting(categorized: CategorizedImport[]): ImportError[] {\n const errors: ImportError[] = []\n const namedImports = categorized.filter(c => c.group === 'named')\n\n for (const {declaration} of namedImports) {\n const specifiers = getNamedSpecifiers(declaration)\n if (specifiers.length > 1 && !areSpecifiersSorted(specifiers)) {\n errors.push({\n node: declaration,\n messageId: 'sortedNames',\n })\n }\n }\n\n return errors\n}\n","import {compare} from '../../lib/compare'\nimport {getSpecifierName} from './getSpecifierName'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function sortSpecifiersText(\n specifiers: TSESTree.ImportSpecifier[],\n sourceCode: {getText: (node: TSESTree.ImportSpecifier) => string},\n): string {\n const sorted = [...specifiers].sort((a, b) => {\n const nameA = getSpecifierName(a)\n const nameB = getSpecifierName(b)\n return compare(nameA, nameB)\n })\n return sorted.map(s => sourceCode.getText(s)).join(', ')\n}\n","import {areSpecifiersSorted} from '../areSpecifiersSorted'\nimport {getNamedSpecifiers} from '../getNamedSpecifiers'\nimport {sortSpecifiersText} from '../sortSpecifiersText'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function formatNamedImport(\n declaration: TSESTree.ImportDeclaration,\n sourceCode: {getText: (node?: unknown) => string},\n): string {\n const specifiers = getNamedSpecifiers(declaration)\n\n if (specifiers.length > 1 && !areSpecifiersSorted(specifiers)) {\n const sortedSpecifiers = sortSpecifiersText(specifiers, sourceCode)\n const source = declaration.source.value\n const prefix = declaration.importKind === 'type' ? 'import type ' : 'import '\n return `${prefix}{${sortedSpecifiers}} from '${source}'`\n }\n\n return sourceCode.getText(declaration)\n}\n","import {formatNamedImport} from './formatNamedImport'\nimport {importGroupOrder} from '../ImportGroupOrder'\nimport type {CategorizedImport} from '../CategorizedImport'\nimport type {ImportGroup} from '../ImportGroup'\n\nexport function buildSortedCode(\n grouped: Record<ImportGroup, CategorizedImport[]>,\n sourceCode: {getText: (node?: unknown) => string},\n): string[] {\n const sortedCode: string[] = []\n\n for (const group of importGroupOrder) {\n for (const {declaration} of grouped[group]) {\n if (group === 'named' || group === 'type')\n sortedCode.push(formatNamedImport(declaration, sourceCode))\n else\n sortedCode.push(sourceCode.getText(declaration))\n }\n }\n\n return sortedCode\n}\n","import type {CategorizedImport} from '../CategorizedImport'\nimport type {ImportGroup} from '../ImportGroup'\n\nexport function groupImportsByType(\n categorized: CategorizedImport[],\n): Record<ImportGroup, CategorizedImport[]> {\n const grouped: Record<ImportGroup, CategorizedImport[]> = {\n 'side-effect': [],\n default: [],\n named: [],\n type: [],\n }\n\n for (const item of categorized)\n grouped[item.group].push(item)\n\n return grouped\n}\n","import {compare} from '../../../lib/compare'\nimport type {CategorizedImport} from '../CategorizedImport'\nimport type {ImportGroup} from '../ImportGroup'\n\nexport function sortImportGroups(\n grouped: Record<ImportGroup, CategorizedImport[]>,\n): void {\n grouped['side-effect'].sort((a, b) => compare(a.sortKey, b.sortKey))\n grouped['default'].sort((a, b) => compare(a.sortKey, b.sortKey))\n grouped['named'].sort((a, b) => compare(a.sortKey, b.sortKey))\n grouped['type'].sort((a, b) => compare(a.sortKey, b.sortKey))\n}\n","import {buildSortedCode} from './buildSortedCode'\nimport {categorizeImports} from '../categorizeImports'\nimport {groupImportsByType} from './groupImportsByType'\nimport {sortImportGroups} from './sortImportGroups'\nimport type {Rule} from 'eslint'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nfunction createFixForGroup(\n fixer: Rule.RuleFixer,\n importDeclarations: TSESTree.ImportDeclaration[],\n sourceCode: {getText: (node?: unknown) => string},\n): Rule.Fix | null {\n if (importDeclarations.length === 0) {\n return null\n }\n\n const categorized = categorizeImports(importDeclarations)\n const grouped = groupImportsByType(categorized)\n\n sortImportGroups(grouped)\n\n const sortedCode = buildSortedCode(grouped, sourceCode)\n .join('\\n')\n\n const firstImport = importDeclarations[0]\n const lastImport = importDeclarations[importDeclarations.length - 1]\n\n return fixer.replaceTextRange(\n [firstImport.range![0], lastImport.range![1]],\n sortedCode,\n )\n}\n\nexport function createFix(\n fixer: Rule.RuleFixer,\n importGroups: TSESTree.ImportDeclaration[][],\n sourceCode: {getText: (node?: unknown) => string},\n): Rule.Fix[] {\n const fixes: Rule.Fix[] = []\n\n for (const group of importGroups) {\n const fix = createFixForGroup(fixer, group, sourceCode)\n if (fix)\n fixes.push(fix)\n }\n\n return fixes\n}\n","import type {TSESTree} from '@typescript-eslint/types'\n\nexport function getImportGroups(programBody: TSESTree.ProgramStatement[]): TSESTree.ImportDeclaration[][] {\n const groups: TSESTree.ImportDeclaration[][] = []\n let currentGroup: TSESTree.ImportDeclaration[] = []\n\n for (const statement of programBody) {\n if (statement.type === 'ImportDeclaration') {\n currentGroup.push(statement)\n continue\n }\n\n if (currentGroup.length > 0) {\n groups.push(currentGroup)\n currentGroup = []\n }\n }\n\n if (currentGroup.length > 0)\n groups.push(currentGroup)\n\n return groups\n}\n","import {categorizeImports} from './categorizeImports'\nimport {checkAlphabeticalSorting} from './checkAlphabeticalSorting'\nimport {checkGroupOrdering} from './checkGroupOrdering'\nimport {checkSpecifiersSorting} from './checkSpecifiersSorting'\nimport {createFix} from './createFix'\nimport {getImportGroups} from './getImportGroups'\nimport type {ImportError} from './ImportError'\nimport type {Rule} from 'eslint'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport const sortedImports: Rule.RuleModule = {\n meta: {\n docs: {\n description: 'Enforce sorted imports alphabetically',\n recommended: true,\n },\n fixable: 'code',\n messages: {\n sortedImports: 'Imports should be sorted alphabetically',\n sortedNames: 'Named imports should be sorted alphabetically',\n wrongGroup: 'Import is in wrong group',\n },\n schema: [],\n type: 'suggestion',\n },\n create(context) {\n return {\n Program(node) {\n const body = node.body as TSESTree.ProgramStatement[]\n const importGroups = getImportGroups(body)\n if (importGroups.length === 0)\n return\n\n const allErrors: ImportError[] = []\n\n // Check each import group independently\n for (const group of importGroups) {\n const categorized = categorizeImports(group)\n const errors: ImportError[] = [\n ...checkGroupOrdering(categorized),\n ...checkAlphabeticalSorting(categorized),\n ...checkSpecifiersSorting(categorized),\n ]\n allErrors.push(...errors)\n }\n\n for (const error of allErrors) {\n context.report({\n node: error.node,\n messageId: error.messageId,\n fix(fixer) {\n const sourceCode = context.sourceCode\n return createFix(fixer, importGroups, sourceCode)\n },\n })\n }\n },\n }\n },\n}\n","import type {ReExportGroup} from './ReExportGroup'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function categorizeReExport(\n declaration: TSESTree.ExportNamedDeclaration | TSESTree.ExportAllDeclaration,\n): ReExportGroup {\n // Example: export * from 'module'\n if (declaration.type === 'ExportAllDeclaration')\n return 're-export-all'\n\n // Example: export type {Type} from 'module'\n if (declaration.exportKind === 'type')\n return 're-export-type'\n\n // Example: export {value} from 'module'\n return 're-export-named'\n}\n","import type {TSESTree} from '@typescript-eslint/types'\n\nexport function getSortKey(\n declaration: TSESTree.ExportNamedDeclaration | TSESTree.ExportAllDeclaration,\n): string {\n if (declaration.type === 'ExportAllDeclaration')\n return declaration.source.value\n\n const specifier = declaration.specifiers[0]\n if (!specifier)\n return ''\n\n return specifier.local.type === 'Identifier'\n ? specifier.local.name\n : specifier.local.value\n}\n","import {categorizeReExport} from './categorizeReExport'\nimport {getSortKey} from './getSortKey'\nimport {ReExportDeclaration} from './ReExportDeclaration'\nimport type {CategorizedReExport} from './CategorizedReExport'\n\nexport function categorizeReExports(declarations: ReExportDeclaration[]): CategorizedReExport[] {\n return declarations.map(declaration => {\n return {\n declaration,\n group: categorizeReExport(declaration),\n sortKey: getSortKey(declaration),\n } as CategorizedReExport\n })\n}\n","import {ReExportGroup} from './ReExportGroup'\n\nexport const reExportGroupOrder: ReExportGroup[] = [\n 're-export-all',\n 're-export-named',\n 're-export-type',\n]\n","import {compare} from '../../lib/compare'\nimport {reExportGroupOrder} from './ReExportGroupOrder'\nimport type {CategorizedReExport} from './CategorizedReExport'\nimport type {ReExportError} from './ReExportError'\n\nexport function checkAlphabeticalSorting(categorized: CategorizedReExport[]): ReExportError[] {\n const errors: ReExportError[] = []\n\n for (const group of reExportGroupOrder) {\n const groupReExports = categorized.filter(c => c.group === group)\n const sorted = [...groupReExports].sort((a, b) =>\n compare(a.sortKey, b.sortKey),\n )\n for (let i = 0; i < groupReExports.length; i++) {\n if (groupReExports[i] !== sorted[i]) {\n errors.push({\n node: groupReExports[i].declaration,\n messageId: 'sortedReExports',\n })\n }\n }\n }\n\n return errors\n}\n","import {reExportGroupOrder} from './ReExportGroupOrder'\nimport type {CategorizedReExport} from './CategorizedReExport'\nimport type {ReExportError} from './ReExportError'\n\nexport function checkGroupOrdering(categorized: CategorizedReExport[]): ReExportError[] {\n const errors: ReExportError[] = []\n\n let currentGroupIndex = -1\n for (const {declaration, group} of categorized) {\n const groupIndex = reExportGroupOrder.indexOf(group)\n if (groupIndex < currentGroupIndex) {\n errors.push({\n node: declaration,\n messageId: 'wrongGroup',\n })\n } else\n currentGroupIndex = groupIndex\n }\n\n return errors\n}\n","import type {TSESTree} from '@typescript-eslint/types'\n\nexport function getSpecifierName(specifier: TSESTree.ExportSpecifier): string {\n return specifier.local.type === 'Identifier'\n ? specifier.local.name\n : String(specifier.local.value)\n}\n","import {compare} from '../../lib/compare'\nimport {getSpecifierName} from './getSpecifierName'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function areSpecifiersSorted(specifiers: TSESTree.ExportSpecifier[]): boolean {\n const names = specifiers.map(s => getSpecifierName(s))\n const sorted = [...names].sort((a, b) => compare(a, b))\n return names.every((name, i) => name === sorted[i])\n}\n","import type {TSESTree} from '@typescript-eslint/types'\n\nexport function getNamedSpecifiers(\n declaration: TSESTree.ExportNamedDeclaration,\n): TSESTree.ExportSpecifier[] {\n return declaration.specifiers.filter(\n s => s.type === 'ExportSpecifier' && s.local.type === 'Identifier',\n ) as TSESTree.ExportSpecifier[]\n}\n","import {CategorizedNamedReExport} from './CategorizedNamedReExport'\nimport type {CategorizedReExport} from './CategorizedReExport'\n\nexport function isNamedReExport(x: CategorizedReExport): x is CategorizedNamedReExport {\n return x.group !== 're-export-all'\n}\n","import {areSpecifiersSorted} from './areSpecifiersSorted'\nimport {getNamedSpecifiers} from './getNamedSpecifiers'\nimport {isNamedReExport} from './isNamedReExport'\nimport type {CategorizedReExport} from './CategorizedReExport'\nimport type {ReExportError} from './ReExportError'\n\nexport function checkSpecifiersSorting(categorized: CategorizedReExport[]): ReExportError[] {\n const errors: ReExportError[] = []\n const namedReExports = categorized.filter(isNamedReExport)\n\n for (const {declaration} of namedReExports) {\n const specifiers = getNamedSpecifiers(declaration)\n const isSorted = areSpecifiersSorted(specifiers)\n if (specifiers.length > 1 && !isSorted) {\n errors.push({\n node: declaration,\n messageId: 'sortedNames',\n })\n }\n }\n\n return errors\n}\n","import {compare} from '../../lib/compare'\nimport {getSpecifierName} from './getSpecifierName'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function sortSpecifiersText(\n specifiers: TSESTree.ExportSpecifier[],\n sourceCode: {getText: (node?: unknown) => string},\n): string {\n const sorted = [...specifiers].sort((a, b) => {\n const nameA = getSpecifierName(a)\n const nameB = getSpecifierName(b)\n return compare(nameA, nameB)\n })\n return sorted.map(s => sourceCode.getText(s)).join(', ')\n}\n","import {areSpecifiersSorted} from '../areSpecifiersSorted'\nimport {getNamedSpecifiers} from '../getNamedSpecifiers'\nimport {sortSpecifiersText} from '../sortSpecifiersText'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport function formatNamedReExport(\n declaration: TSESTree.ExportNamedDeclaration,\n sourceCode: {getText: (node?: unknown) => string},\n): string {\n const specifiers = getNamedSpecifiers(declaration)\n\n if (specifiers.length > 1 && !areSpecifiersSorted(specifiers)) {\n const sortedSpecifiers = sortSpecifiersText(specifiers, sourceCode)\n const source = declaration.source!.value\n const prefix = declaration.exportKind === 'type' ? 'export type ' : 'export '\n return `${prefix}{${sortedSpecifiers}} from '${source}'`\n }\n\n return sourceCode.getText(declaration)\n}\n","import {formatNamedReExport} from './formatNamedReExport'\nimport {isNamedReExport} from '../isNamedReExport'\nimport {reExportGroupOrder} from '../ReExportGroupOrder'\nimport type {CategorizedReExport} from '../CategorizedReExport'\nimport type {ReExportGroup} from '../ReExportGroup'\n\nexport function buildSortedCode(\n grouped: Record<ReExportGroup, CategorizedReExport[]>,\n sourceCode: {getText: (node?: unknown) => string},\n): string[] {\n const sortedCode: string[] = []\n\n for (const group of reExportGroupOrder) {\n for (const item of grouped[group]) {\n if (isNamedReExport(item)) {\n sortedCode.push(\n formatNamedReExport(\n item.declaration,\n sourceCode,\n ),\n )\n } else\n sortedCode.push(sourceCode.getText(item.declaration))\n }\n }\n\n return sortedCode\n}\n","import type {CategorizedReExport} from '../CategorizedReExport'\nimport type {ReExportGroup} from '../ReExportGroup'\n\nexport function groupReExportsByType(\n categorized: CategorizedReExport[],\n): Record<ReExportGroup, CategorizedReExport[]> {\n const grouped: Record<ReExportGroup, CategorizedReExport[]> = {\n 're-export-all': [],\n 're-export-named': [],\n 're-export-type': [],\n }\n\n for (const item of categorized)\n grouped[item.group].push(item)\n\n return grouped\n}\n","import {compare} from '../../../lib/compare'\nimport type {CategorizedReExport} from '../CategorizedReExport'\nimport type {ReExportGroup} from '../ReExportGroup'\n\nexport function sortExportGroups(\n grouped: Record<ReExportGroup, CategorizedReExport[]>,\n): void {\n grouped['re-export-all'].sort((a, b) => compare(a.sortKey, b.sortKey))\n grouped['re-export-named'].sort((a, b) => compare(a.sortKey, b.sortKey))\n grouped['re-export-type'].sort((a, b) => compare(a.sortKey, b.sortKey))\n}\n","import {buildSortedCode} from './buildSortedCode'\nimport {categorizeReExports} from '../categorizeReExports'\nimport {groupReExportsByType} from './groupReExportsByType'\nimport {ReExportDeclaration} from '../ReExportDeclaration'\nimport {sortExportGroups} from './sortExportGroups'\nimport type {Rule} from 'eslint'\n\nfunction createFixForGroup(\n fixer: Rule.RuleFixer,\n reExportDeclarations: ReExportDeclaration[],\n sourceCode: {getText: (node?: unknown) => string},\n): Rule.Fix | null {\n if (reExportDeclarations.length === 0) {\n return null\n }\n\n const categorized = categorizeReExports(reExportDeclarations)\n const grouped = groupReExportsByType(categorized)\n\n sortExportGroups(grouped)\n\n const sortedCode = buildSortedCode(grouped, sourceCode)\n .join('\\n')\n\n const firstReExport = reExportDeclarations[0]\n const lastReExport = reExportDeclarations[reExportDeclarations.length - 1]\n\n return fixer.replaceTextRange(\n [firstReExport.range![0], lastReExport.range![1]],\n sortedCode,\n )\n}\n\nexport function createFix(\n fixer: Rule.RuleFixer,\n reExportGroups: ReExportDeclaration[][],\n sourceCode: {getText: (node?: unknown) => string},\n): Rule.Fix[] {\n const fixes: Rule.Fix[] = []\n\n for (const group of reExportGroups) {\n const fix = createFixForGroup(fixer, group, sourceCode)\n if (fix)\n fixes.push(fix)\n }\n\n return fixes\n}\n","import type {ReExportDeclaration} from './ReExportDeclaration'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nfunction isReExportDeclaration(\n statement: TSESTree.ProgramStatement,\n): statement is ReExportDeclaration {\n return (statement.type === 'ExportNamedDeclaration' && statement.source !== null)\n || statement.type === 'ExportAllDeclaration'\n}\n\nexport function getReExportGroups(\n programBody: TSESTree.ProgramStatement[],\n): ReExportDeclaration[][] {\n const groups: ReExportDeclaration[][] = []\n let currentGroup: ReExportDeclaration[] = []\n\n for (const statement of programBody) {\n if (isReExportDeclaration(statement)) {\n currentGroup.push(statement)\n continue\n }\n\n if (currentGroup.length > 0) {\n groups.push(currentGroup)\n currentGroup = []\n }\n }\n\n if (currentGroup.length > 0)\n groups.push(currentGroup)\n\n return groups\n}\n","import {categorizeReExports} from './categorizeReExports'\nimport {checkAlphabeticalSorting} from './checkAlphabeticalSorting'\nimport {checkGroupOrdering} from './checkGroupOrdering'\nimport {checkSpecifiersSorting} from './checkSpecifiersSorting'\nimport {createFix} from './createFix'\nimport {getReExportGroups} from './getReExportGroups'\nimport type {ReExportError} from './ReExportError'\nimport type {Rule} from 'eslint'\nimport type {TSESTree} from '@typescript-eslint/types'\n\nexport const sortedReExports: Rule.RuleModule = {\n meta: {\n docs: {\n description: 'Enforce sorted exports alphabetically',\n recommended: true,\n },\n fixable: 'code',\n messages: {\n sortedReExports: 'Exports should be sorted alphabetically',\n sortedNames: 'Named exports should be sorted alphabetically',\n wrongGroup: 'Export is in wrong group',\n },\n schema: [],\n type: 'suggestion',\n },\n create(context) {\n return {\n Program(node) {\n const body = node.body as TSESTree.ProgramStatement[]\n const reExportGroups = getReExportGroups(body)\n if (reExportGroups.length === 0)\n return\n\n const allErrors: ReExportError[] = []\n\n // Check each re-export group independently\n for (const group of reExportGroups) {\n const categorized = categorizeReExports(group)\n const errors: ReExportError[] = [\n ...checkGroupOrdering(categorized),\n ...checkAlphabeticalSorting(categorized),\n ...checkSpecifiersSorting(categorized),\n ]\n allErrors.push(...errors)\n }\n\n for (const error of allErrors) {\n context.report({\n node: error.node,\n messageId: error.messageId,\n fix(fixer) {\n const sourceCode = context.sourceCode\n return createFix(fixer, reExportGroups, sourceCode)\n },\n })\n }\n },\n }\n },\n}\n"],"mappings":"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","names":["getSortKey","getSortKey","checkAlphabeticalSorting","checkGroupOrdering","getSpecifierName","areSpecifiersSorted","getSpecifierName","getNamedSpecifiers","checkSpecifiersSorting","getNamedSpecifiers","areSpecifiersSorted","sortSpecifiersText","getSpecifierName","getNamedSpecifiers","areSpecifiersSorted","sortSpecifiersText","buildSortedCode","createFixForGroup","buildSortedCode","createFix","checkGroupOrdering","checkAlphabeticalSorting","checkSpecifiersSorting","createFix"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@borela-tech/eslint-config",
3
- "version": "2.1.0",
3
+ "version": "2.1.2",
4
4
  "description": "ESLint config used in Borela Tech projects.",
5
5
  "scripts": {
6
6
  "build": "./bin/build",
@@ -55,28 +55,6 @@ ruleTester.run('imports-and-re-exports-at-top', importsAndReExportsAtTop, {
55
55
  code: '',
56
56
  }],
57
57
  invalid: [{
58
- code: dedent`
59
- export {a} from 'aaa'
60
- import {b} from 'bbb'
61
- const c = 1
62
- `,
63
- errors: [{messageId: 'importsAndReExportsAtTop'}],
64
- output: dedent`
65
- import {b} from 'bbb'
66
- export {a} from 'aaa'
67
- const c = 1
68
- `,
69
- }, {
70
- code: dedent`
71
- export * from 'aaa'
72
- import {b} from 'bbb'
73
- `,
74
- errors: [{messageId: 'importsAndReExportsAtTop'}],
75
- output: dedent`
76
- import {b} from 'bbb'
77
- export * from 'aaa'
78
- `,
79
- }, {
80
58
  code: dedent`
81
59
  export {z} from 'zzz'
82
60
  const c = 1
@@ -104,15 +82,5 @@ ruleTester.run('imports-and-re-exports-at-top', importsAndReExportsAtTop, {
104
82
  export {c} from 'ccc'
105
83
  const a = 1
106
84
  `,
107
- }, {
108
- code: dedent`
109
- export type {a} from 'aaa'
110
- import {b} from 'bbb'
111
- `,
112
- errors: [{messageId: 'importsAndReExportsAtTop'}],
113
- output: dedent`
114
- import {b} from 'bbb'
115
- export type {a} from 'aaa'
116
- `,
117
85
  }],
118
86
  })
@@ -56,6 +56,49 @@ ruleTester.run('sorted-imports', sortedImports, {
56
56
  output: dedent`
57
57
  import {a, b, c} from 'bar'
58
58
  `,
59
+ }, {
60
+ code: dedent`
61
+ import {b} from 'b'
62
+ const x = 1
63
+ import {c} from 'c'
64
+ import {a} from 'a'
65
+ `,
66
+ errors: [{messageId: 'sortedImports'}, {messageId: 'sortedImports'}],
67
+ output: dedent`
68
+ import {b} from 'b'
69
+ const x = 1
70
+ import {a} from 'a'
71
+ import {c} from 'c'
72
+ `,
73
+ }, {
74
+ code: dedent`
75
+ import {b} from 'b'
76
+ import {a} from 'a'
77
+ const x = 1
78
+ import {d} from 'd'
79
+ import {c} from 'c'
80
+ const y = 2
81
+ import {f} from 'f'
82
+ import {e} from 'e'
83
+ `,
84
+ errors: [
85
+ {messageId: 'sortedImports'},
86
+ {messageId: 'sortedImports'},
87
+ {messageId: 'sortedImports'},
88
+ {messageId: 'sortedImports'},
89
+ {messageId: 'sortedImports'},
90
+ {messageId: 'sortedImports'},
91
+ ],
92
+ output: dedent`
93
+ import {a} from 'a'
94
+ import {b} from 'b'
95
+ const x = 1
96
+ import {c} from 'c'
97
+ import {d} from 'd'
98
+ const y = 2
99
+ import {e} from 'e'
100
+ import {f} from 'f'
101
+ `,
59
102
  }, {
60
103
  code: dedent`
61
104
  import {z, a} from 'bar'
@@ -89,6 +89,49 @@ ruleTester.run('sorted-re-exports', sortedReExports, {
89
89
  output: dedent`
90
90
  export {a, b, c} from 'bar'
91
91
  `,
92
+ }, {
93
+ code: dedent`
94
+ export {b} from 'b'
95
+ const x = 1
96
+ export {c} from 'c'
97
+ export {a} from 'a'
98
+ `,
99
+ errors: [{messageId: 'sortedReExports'}, {messageId: 'sortedReExports'}],
100
+ output: dedent`
101
+ export {b} from 'b'
102
+ const x = 1
103
+ export {a} from 'a'
104
+ export {c} from 'c'
105
+ `,
106
+ }, {
107
+ code: dedent`
108
+ export {b} from 'b'
109
+ export {a} from 'a'
110
+ const x = 1
111
+ export {d} from 'd'
112
+ export {c} from 'c'
113
+ const y = 2
114
+ export {f} from 'f'
115
+ export {e} from 'e'
116
+ `,
117
+ errors: [
118
+ {messageId: 'sortedReExports'},
119
+ {messageId: 'sortedReExports'},
120
+ {messageId: 'sortedReExports'},
121
+ {messageId: 'sortedReExports'},
122
+ {messageId: 'sortedReExports'},
123
+ {messageId: 'sortedReExports'},
124
+ ],
125
+ output: dedent`
126
+ export {a} from 'a'
127
+ export {b} from 'b'
128
+ const x = 1
129
+ export {c} from 'c'
130
+ export {d} from 'd'
131
+ const y = 2
132
+ export {e} from 'e'
133
+ export {f} from 'f'
134
+ `,
92
135
  }, {
93
136
  code: dedent`
94
137
  export {z, a} from 'bar'
@@ -1,5 +1,5 @@
1
1
  export interface StatementIndices {
2
- firstImport: number
3
- firstReExport: number
4
- firstOther: number
2
+ firstRegularStatement: number
3
+ lastImport: number
4
+ lastReExport: number
5
5
  }
@@ -0,0 +1,25 @@
1
+ import {getStatementType} from './getStatementType'
2
+ import type {StatementIndices} from './StatementIndices'
3
+ import type {StatementType} from './statementType'
4
+ import type {TSESTree} from '@typescript-eslint/types'
5
+
6
+ export function findStatementIndices(
7
+ statements: TSESTree.Statement[],
8
+ ): StatementIndices {
9
+ let firstRegularStatement = -1
10
+ let lastImport = -1
11
+ let lastReExport = -1
12
+
13
+ for (let i = 0; i < statements.length; i++) {
14
+ const type: StatementType = getStatementType(statements[i])
15
+
16
+ if (type === 'import')
17
+ lastImport = i
18
+ else if (type === 're-export')
19
+ lastReExport = i
20
+ else if (type === 'other' && firstRegularStatement === -1)
21
+ firstRegularStatement = i
22
+ }
23
+
24
+ return {firstRegularStatement, lastImport, lastReExport}
25
+ }
@@ -5,25 +5,21 @@ export function hasViolation(
5
5
  indices: StatementIndices,
6
6
  categories: CategorizedStatements,
7
7
  ): boolean {
8
- const {
9
- firstImport,
10
- firstReExport,
11
- firstOther,
12
- } = indices
8
+ const {firstRegularStatement, lastImport, lastReExport} = indices
13
9
 
14
- // No imports or no re-exports.
15
- if (categories.imports.length === 0 || categories.reExports.length === 0)
10
+ if (categories.imports.length === 0 && categories.reExports.length === 0)
16
11
  return false
17
12
 
18
- const firstImportOrReExport = Math.min(firstImport, firstReExport)
19
- const hasOtherBeforeImportOrReExport =
20
- firstOther !== -1 && firstOther < firstImportOrReExport
13
+ const hasImportAfterRegularStatement = (
14
+ categories.imports.length > 0
15
+ && firstRegularStatement !== -1
16
+ && lastImport > firstRegularStatement
17
+ )
18
+ const hasReExportAfterRegularStatement = (
19
+ categories.reExports.length > 0
20
+ && firstRegularStatement !== -1
21
+ && lastReExport > firstRegularStatement
22
+ )
21
23
 
22
- // Violation if:
23
- // 1. Other statements appear before imports/re-exports.
24
- // 2. Re-exports appear before imports.
25
- if (hasOtherBeforeImportOrReExport || firstImport > firstReExport)
26
- return true
27
-
28
- return false
24
+ return hasImportAfterRegularStatement || hasReExportAfterRegularStatement
29
25
  }
@@ -1,5 +1,5 @@
1
1
  import {categorizeStatements} from './categorizeStatements'
2
- import {findFirstIndices} from './findFirstIndices'
2
+ import {findStatementIndices} from './findStatementIndices'
3
3
  import {generateSortedText} from './generateSortedText'
4
4
  import {hasViolation} from './hasViolation'
5
5
  import type {Rule} from 'eslint'
@@ -25,7 +25,7 @@ export const importsAndReExportsAtTop: Rule.RuleModule = {
25
25
  Program(node) {
26
26
  const statements = node.body as TSESTree.Statement[]
27
27
  const categories = categorizeStatements(statements)
28
- const indices = findFirstIndices(statements)
28
+ const indices = findStatementIndices(statements)
29
29
 
30
30
  if (!hasViolation(indices, categories))
31
31
  return
@@ -1,18 +1,19 @@
1
1
  import {buildSortedCode} from './buildSortedCode'
2
2
  import {categorizeImports} from '../categorizeImports'
3
- import {getReplacementRange} from './getReplacementRange'
4
3
  import {groupImportsByType} from './groupImportsByType'
5
4
  import {sortImportGroups} from './sortImportGroups'
6
5
  import type {Rule} from 'eslint'
7
6
  import type {TSESTree} from '@typescript-eslint/types'
8
7
 
9
- export function createFix(
8
+ function createFixForGroup(
10
9
  fixer: Rule.RuleFixer,
11
10
  importDeclarations: TSESTree.ImportDeclaration[],
12
11
  sourceCode: {getText: (node?: unknown) => string},
13
- programBody: TSESTree.ProgramStatement[],
14
- ) {
15
- const range = getReplacementRange(programBody)
12
+ ): Rule.Fix | null {
13
+ if (importDeclarations.length === 0) {
14
+ return null
15
+ }
16
+
16
17
  const categorized = categorizeImports(importDeclarations)
17
18
  const grouped = groupImportsByType(categorized)
18
19
 
@@ -21,8 +22,27 @@ export function createFix(
21
22
  const sortedCode = buildSortedCode(grouped, sourceCode)
22
23
  .join('\n')
23
24
 
25
+ const firstImport = importDeclarations[0]
26
+ const lastImport = importDeclarations[importDeclarations.length - 1]
27
+
24
28
  return fixer.replaceTextRange(
25
- [range.start, range.end],
29
+ [firstImport.range![0], lastImport.range![1]],
26
30
  sortedCode,
27
31
  )
28
32
  }
33
+
34
+ export function createFix(
35
+ fixer: Rule.RuleFixer,
36
+ importGroups: TSESTree.ImportDeclaration[][],
37
+ sourceCode: {getText: (node?: unknown) => string},
38
+ ): Rule.Fix[] {
39
+ const fixes: Rule.Fix[] = []
40
+
41
+ for (const group of importGroups) {
42
+ const fix = createFixForGroup(fixer, group, sourceCode)
43
+ if (fix)
44
+ fixes.push(fix)
45
+ }
46
+
47
+ return fixes
48
+ }
@@ -0,0 +1,23 @@
1
+ import type {TSESTree} from '@typescript-eslint/types'
2
+
3
+ export function getImportGroups(programBody: TSESTree.ProgramStatement[]): TSESTree.ImportDeclaration[][] {
4
+ const groups: TSESTree.ImportDeclaration[][] = []
5
+ let currentGroup: TSESTree.ImportDeclaration[] = []
6
+
7
+ for (const statement of programBody) {
8
+ if (statement.type === 'ImportDeclaration') {
9
+ currentGroup.push(statement)
10
+ continue
11
+ }
12
+
13
+ if (currentGroup.length > 0) {
14
+ groups.push(currentGroup)
15
+ currentGroup = []
16
+ }
17
+ }
18
+
19
+ if (currentGroup.length > 0)
20
+ groups.push(currentGroup)
21
+
22
+ return groups
23
+ }
@@ -3,7 +3,7 @@ import {checkAlphabeticalSorting} from './checkAlphabeticalSorting'
3
3
  import {checkGroupOrdering} from './checkGroupOrdering'
4
4
  import {checkSpecifiersSorting} from './checkSpecifiersSorting'
5
5
  import {createFix} from './createFix'
6
- import {getImportDeclarations} from './getImportDeclarations'
6
+ import {getImportGroups} from './getImportGroups'
7
7
  import type {ImportError} from './ImportError'
8
8
  import type {Rule} from 'eslint'
9
9
  import type {TSESTree} from '@typescript-eslint/types'
@@ -27,24 +27,30 @@ export const sortedImports: Rule.RuleModule = {
27
27
  return {
28
28
  Program(node) {
29
29
  const body = node.body as TSESTree.ProgramStatement[]
30
- const declarations = getImportDeclarations(body)
31
- if (declarations.length === 0)
30
+ const importGroups = getImportGroups(body)
31
+ if (importGroups.length === 0)
32
32
  return
33
33
 
34
- const categorized = categorizeImports(declarations)
35
- const errors: ImportError[] = [
36
- ...checkGroupOrdering(categorized),
37
- ...checkAlphabeticalSorting(categorized),
38
- ...checkSpecifiersSorting(categorized),
39
- ]
34
+ const allErrors: ImportError[] = []
40
35
 
41
- for (const error of errors) {
36
+ // Check each import group independently
37
+ for (const group of importGroups) {
38
+ const categorized = categorizeImports(group)
39
+ const errors: ImportError[] = [
40
+ ...checkGroupOrdering(categorized),
41
+ ...checkAlphabeticalSorting(categorized),
42
+ ...checkSpecifiersSorting(categorized),
43
+ ]
44
+ allErrors.push(...errors)
45
+ }
46
+
47
+ for (const error of allErrors) {
42
48
  context.report({
43
49
  node: error.node,
44
50
  messageId: error.messageId,
45
51
  fix(fixer) {
46
52
  const sourceCode = context.sourceCode
47
- return createFix(fixer, declarations, sourceCode, body)
53
+ return createFix(fixer, importGroups, sourceCode)
48
54
  },
49
55
  })
50
56
  }
@@ -1,19 +1,19 @@
1
1
  import {buildSortedCode} from './buildSortedCode'
2
2
  import {categorizeReExports} from '../categorizeReExports'
3
- import {getReplacementRange} from './getReplacementRange'
4
3
  import {groupReExportsByType} from './groupReExportsByType'
5
4
  import {ReExportDeclaration} from '../ReExportDeclaration'
6
5
  import {sortExportGroups} from './sortExportGroups'
7
6
  import type {Rule} from 'eslint'
8
- import type {TSESTree} from '@typescript-eslint/types'
9
7
 
10
- export function createFix(
8
+ function createFixForGroup(
11
9
  fixer: Rule.RuleFixer,
12
10
  reExportDeclarations: ReExportDeclaration[],
13
11
  sourceCode: {getText: (node?: unknown) => string},
14
- programBody: TSESTree.ProgramStatement[],
15
- ) {
16
- const range = getReplacementRange(programBody)
12
+ ): Rule.Fix | null {
13
+ if (reExportDeclarations.length === 0) {
14
+ return null
15
+ }
16
+
17
17
  const categorized = categorizeReExports(reExportDeclarations)
18
18
  const grouped = groupReExportsByType(categorized)
19
19
 
@@ -22,8 +22,27 @@ export function createFix(
22
22
  const sortedCode = buildSortedCode(grouped, sourceCode)
23
23
  .join('\n')
24
24
 
25
+ const firstReExport = reExportDeclarations[0]
26
+ const lastReExport = reExportDeclarations[reExportDeclarations.length - 1]
27
+
25
28
  return fixer.replaceTextRange(
26
- [range.start, range.end],
29
+ [firstReExport.range![0], lastReExport.range![1]],
27
30
  sortedCode,
28
31
  )
29
32
  }
33
+
34
+ export function createFix(
35
+ fixer: Rule.RuleFixer,
36
+ reExportGroups: ReExportDeclaration[][],
37
+ sourceCode: {getText: (node?: unknown) => string},
38
+ ): Rule.Fix[] {
39
+ const fixes: Rule.Fix[] = []
40
+
41
+ for (const group of reExportGroups) {
42
+ const fix = createFixForGroup(fixer, group, sourceCode)
43
+ if (fix)
44
+ fixes.push(fix)
45
+ }
46
+
47
+ return fixes
48
+ }
@@ -0,0 +1,33 @@
1
+ import type {ReExportDeclaration} from './ReExportDeclaration'
2
+ import type {TSESTree} from '@typescript-eslint/types'
3
+
4
+ function isReExportDeclaration(
5
+ statement: TSESTree.ProgramStatement,
6
+ ): statement is ReExportDeclaration {
7
+ return (statement.type === 'ExportNamedDeclaration' && statement.source !== null)
8
+ || statement.type === 'ExportAllDeclaration'
9
+ }
10
+
11
+ export function getReExportGroups(
12
+ programBody: TSESTree.ProgramStatement[],
13
+ ): ReExportDeclaration[][] {
14
+ const groups: ReExportDeclaration[][] = []
15
+ let currentGroup: ReExportDeclaration[] = []
16
+
17
+ for (const statement of programBody) {
18
+ if (isReExportDeclaration(statement)) {
19
+ currentGroup.push(statement)
20
+ continue
21
+ }
22
+
23
+ if (currentGroup.length > 0) {
24
+ groups.push(currentGroup)
25
+ currentGroup = []
26
+ }
27
+ }
28
+
29
+ if (currentGroup.length > 0)
30
+ groups.push(currentGroup)
31
+
32
+ return groups
33
+ }
@@ -3,7 +3,7 @@ import {checkAlphabeticalSorting} from './checkAlphabeticalSorting'
3
3
  import {checkGroupOrdering} from './checkGroupOrdering'
4
4
  import {checkSpecifiersSorting} from './checkSpecifiersSorting'
5
5
  import {createFix} from './createFix'
6
- import {getReExportDeclarations} from './getReExportDeclarations'
6
+ import {getReExportGroups} from './getReExportGroups'
7
7
  import type {ReExportError} from './ReExportError'
8
8
  import type {Rule} from 'eslint'
9
9
  import type {TSESTree} from '@typescript-eslint/types'
@@ -27,24 +27,30 @@ export const sortedReExports: Rule.RuleModule = {
27
27
  return {
28
28
  Program(node) {
29
29
  const body = node.body as TSESTree.ProgramStatement[]
30
- const declarations = getReExportDeclarations(body)
31
- if (declarations.length === 0)
30
+ const reExportGroups = getReExportGroups(body)
31
+ if (reExportGroups.length === 0)
32
32
  return
33
33
 
34
- const categorized = categorizeReExports(declarations)
35
- const errors: ReExportError[] = [
36
- ...checkGroupOrdering(categorized),
37
- ...checkAlphabeticalSorting(categorized),
38
- ...checkSpecifiersSorting(categorized),
39
- ]
34
+ const allErrors: ReExportError[] = []
40
35
 
41
- for (const error of errors) {
36
+ // Check each re-export group independently
37
+ for (const group of reExportGroups) {
38
+ const categorized = categorizeReExports(group)
39
+ const errors: ReExportError[] = [
40
+ ...checkGroupOrdering(categorized),
41
+ ...checkAlphabeticalSorting(categorized),
42
+ ...checkSpecifiersSorting(categorized),
43
+ ]
44
+ allErrors.push(...errors)
45
+ }
46
+
47
+ for (const error of allErrors) {
42
48
  context.report({
43
49
  node: error.node,
44
50
  messageId: error.messageId,
45
51
  fix(fixer) {
46
52
  const sourceCode = context.sourceCode
47
- return createFix(fixer, declarations, sourceCode, body)
53
+ return createFix(fixer, reExportGroups, sourceCode)
48
54
  },
49
55
  })
50
56
  }
@@ -1,25 +0,0 @@
1
- import {getStatementType} from './getStatementType'
2
- import type {StatementIndices} from './StatementIndices'
3
- import type {StatementType} from './statementType'
4
- import type {TSESTree} from '@typescript-eslint/types'
5
-
6
- export function findFirstIndices(
7
- statements: TSESTree.Statement[],
8
- ): StatementIndices {
9
- let firstImport = Infinity
10
- let firstReExport = Infinity
11
- let firstOther = -1
12
-
13
- for (let i = 0; i < statements.length; i++) {
14
- const type: StatementType = getStatementType(statements[i])
15
-
16
- if (type === 'import' && firstImport === Infinity)
17
- firstImport = i
18
- else if (type === 're-export' && firstReExport === Infinity)
19
- firstReExport = i
20
- else if (type === 'other' && firstOther === -1)
21
- firstOther = i
22
- }
23
-
24
- return {firstImport, firstReExport, firstOther}
25
- }
@@ -1,8 +0,0 @@
1
- import type {TSESTree} from '@typescript-eslint/types'
2
-
3
- export function getImportDeclarations(programBody: TSESTree.ProgramStatement[]): TSESTree.ImportDeclaration[] {
4
- return programBody.filter(
5
- (statement): statement is TSESTree.ImportDeclaration =>
6
- statement.type === 'ImportDeclaration',
7
- )
8
- }
@@ -1,12 +0,0 @@
1
- import type {ReExportDeclaration} from './ReExportDeclaration'
2
- import type {TSESTree} from '@typescript-eslint/types'
3
-
4
- export function getReExportDeclarations(
5
- programBody: TSESTree.ProgramStatement[],
6
- ): ReExportDeclaration[] {
7
- return programBody.filter(
8
- (statement): statement is ReExportDeclaration =>
9
- (statement.type === 'ExportNamedDeclaration' && statement.source !== null)
10
- || statement.type === 'ExportAllDeclaration',
11
- )
12
- }