diff --git a/README.md b/README.md index 2873f63..682d856 100644 --- a/README.md +++ b/README.md @@ -254,6 +254,15 @@ import g from "."; import h from "./constants"; import i from "./styles"; +// TypeScript import assignments. +import J = require("../parent"); +import J = require("./sibling"); +export import K = require("an-npm-package"); +import L = require("different-npm-package"); +import M = Namespace; +export import N = Namespace.A.B.C; +import O = Namespace.A.C; + // Different types of exports: export { a } from "../.."; export { b } from "/"; @@ -365,6 +374,8 @@ Side effect imports have `\u0000` _prepended_ to their `from` string (starts wit Type imports have `\u0000` _appended_ to their `from` string (ends with `\u0000`). You can match them with `"\\u0000$"` – but you probably need more than that to avoid them also being matched by other regexes. +TypeScript import assignments have `\u0001` (for `import A = B.C.D`) or `\u0002` (for `import A = require("A")`) prepended to their `from` string (starts with `\u0001` or `\u0002`). It is _not_ possible to distinguish `export import A =` and `import A =`. + All imports that match the same regex are sorted internally as mentioned in [Sort order]. This is the default value for the `groups` option: @@ -384,6 +395,8 @@ This is the default value for the `groups` option: // Relative imports. // Anything that starts with a dot. ["^\\."], + // TypeScript import assignments. + ["^\\u0001", "^\\u0002"], ]; ``` @@ -677,7 +690,7 @@ Use [custom grouping], setting the `groups` option to only have a single inner a For example, here’s the default value but changed to a single inner array: ```js -[["^\\u0000", "^node:", "^@?\\w", "^", "^\\."]]; +[["^\\u0000", "^node:", "^@?\\w", "^", "^\\.", "^\\u0001", "^\\u0002"]]; ``` (By default, each string is in its _own_ array (that’s 5 inner arrays) – causing a blank line between each.) diff --git a/examples/.eslintrc.js b/examples/.eslintrc.js index 44df29b..8ffd10f 100644 --- a/examples/.eslintrc.js +++ b/examples/.eslintrc.js @@ -103,7 +103,7 @@ module.exports = { "error", { // The default grouping, but with no blank lines. - groups: [["^\\u0000", "^node:", "^@?\\w", "^", "^\\."]], + groups: [["^\\u0000", "^node:", "^@?\\w", "^", "^\\.", "^\\u0001", "^\\u0002"]], }, ], }, @@ -115,7 +115,7 @@ module.exports = { "error", { // The default grouping, but in reverse. - groups: [["^\\."], ["^"], ["^@?\\w"], ["^node:"], ["^\\u0000"]], + groups: [["^\\u0001", "^\\u0002"], ["^\\."], ["^"], ["^@?\\w"], ["^node:"], ["^\\u0000"]], }, ], }, @@ -128,7 +128,7 @@ module.exports = { "error", { // The default grouping, but with type imports first as a separate group. - groups: [["^.*\\u0000$"], ["^\\u0000"], ["^node:"], ["^@?\\w"], ["^"], ["^\\."]], + groups: [["^.*\\u0000$"], ["^\\u0000"], ["^node:"], ["^@?\\w"], ["^"], ["^\\."], ["^\\u0001", "^\\u0002"]], }, ], }, @@ -141,7 +141,7 @@ module.exports = { "error", { // The default grouping, but with type imports last as a separate group. - groups: [["^\\u0000"], ["^node:"], ["^@?\\w"], ["^"], ["^\\."], ["^.+\\u0000$"]], + groups: [["^\\u0000"], ["^node:"], ["^@?\\w"], ["^"], ["^\\."], ["^\\u0001", "^\\u0002"], ["^.+\\u0000$"]], }, ], }, @@ -162,6 +162,7 @@ module.exports = { ["^@?\\w"], ["^"], ["^\\."], + ["^\\u0001", "^\\u0002"], ], }, ], @@ -182,6 +183,7 @@ module.exports = { ["^@?\\w"], ["^"], ["^\\."], + ["^\\u0001", "^\\u0002"], ["^node:.*\\u0000$", "^@?\\w.*\\u0000$", "^[^.].*\\u0000$", "^\\..*\\u0000$"], ], }, @@ -202,6 +204,7 @@ module.exports = { ["^@?\\w.*\\u0000$", "^@?\\w"], ["(?<=\\u0000)$", "^"], ["^\\..*\\u0000$", "^\\."], + ["^\\u0001", "^\\u0002"], ], }, ], diff --git a/examples/readme-order.prettier.ts b/examples/readme-order.prettier.ts index f5e9c73..b088097 100644 --- a/examples/readme-order.prettier.ts +++ b/examples/readme-order.prettier.ts @@ -25,6 +25,15 @@ import g from "."; import h from "./constants"; import i from "./styles"; +// TypeScript import assignments. +import J = require("../parent"); +import J = require("./sibling"); +export import K = require("an-npm-package"); +import L = require("different-npm-package"); +import M = Namespace; +export import N = Namespace.A.B.C; +import O = Namespace.A.C; + // Different types of exports: export { a } from "../.."; export { b } from "/"; diff --git a/src/imports.js b/src/imports.js index f7d392a..1dcb6de 100644 --- a/src/imports.js +++ b/src/imports.js @@ -16,9 +16,8 @@ const defaultGroups = [ // Relative imports. // Anything that starts with a dot. ["^\\."], - // Namespace import. - // Anything that starts with an equal sign. - ["^= "], + // TypeScript import assignments. + ["^\\u0001", "^\\u0002"], ]; module.exports = { @@ -100,14 +99,15 @@ function makeSortedItems(items, outerGroups) { for (const item of items) { const { originalSource } = item.source; - const source = item.isSideEffectImport + const sourceWithControlCharacter = item.isSideEffectImport ? `\0${originalSource}` - : item.source.kind !== "value" - ? `${originalSource}\0` - : originalSource; + : getSourceWithControlCharacterFromKind(originalSource, item.source.kind); const [matchedGroup] = shared .flatMap(itemGroups, (groups) => - groups.map((group) => [group, group.regex.exec(source)]) + groups.map((group) => [ + group, + group.regex.exec(sourceWithControlCharacter), + ]) ) .reduce( ([group, longestMatch], [nextGroup, nextMatch]) => @@ -133,6 +133,20 @@ function makeSortedItems(items, outerGroups) { ); } +function getSourceWithControlCharacterFromKind(originalSource, kind) { + switch (kind) { + case shared.KIND_VALUE: + return originalSource; + case shared.KIND_TS_IMPORT_ASSIGNMENT_REQUIRE: + return `\u0001${originalSource}`; + case shared.KIND_TS_IMPORT_ASSIGNMENT_NAMESPACE: + return `\u0002${originalSource}`; + default: + // `type` and `typeof`. + return `${originalSource}\u0000`; + } +} + // Exclude "ImportDefaultSpecifier" – the "def" in `import def, {a, b}`. function getSpecifiers(importNode) { switch (importNode.type) { @@ -167,7 +181,8 @@ function isSideEffectImport(importNode, sourceCode) { default: return ( importNode.specifiers.length === 0 && - (!importNode.importKind || importNode.importKind === "value") && + (!importNode.importKind || + importNode.importKind === shared.KIND_VALUE) && !shared.isPunctuator( sourceCode.getFirstToken(importNode, { skip: 1 }), "{" diff --git a/src/shared.js b/src/shared.js index 4007cbc..2c0123e 100644 --- a/src/shared.js +++ b/src/shared.js @@ -795,72 +795,8 @@ function isNewline(node) { return node.type === "Newline"; } -function getSourceFromTSQualifiedName(sourceCode, node) { - let left; - let right; - switch (node.left.type) { - case "Identifier": { - left = node.left.name; - break; - } - case "TSQualifiedName": { - left = getSourceFromTSQualifiedName(sourceCode, node.left); - break; - } - default: { - left = ``; - break; - } - } - switch (node.right.type) { - case "Identifier": { - right = `.${node.right.name}`; - break; - } - default: { - right = ``; - break; - } - } - return left + right; -} - -function getSourceFromModuleReference(sourceCode, node) { - switch (node.type) { - case "TSExternalModuleReference": { - switch (node.expression.type) { - case "Literal": { - return node.expression.value; - } - default: { - return sourceCode.text.slice(...node.expression.range); - } - } - } - case "TSQualifiedName": { - return `= ${getSourceFromTSQualifiedName(sourceCode, node)}`; - } - case "Identifier": { - return `= ${node.name}`; - } - default: { - return ``; - } - } -} - function getSource(sourceCode, node) { - let source; - switch (node.type) { - case "TSImportEqualsDeclaration": { - source = getSourceFromModuleReference(sourceCode, node.moduleReference); - break; - } - default: { - source = node.source.value; - break; - } - } + const [source, kind] = getSourceTextAndKind(sourceCode, node); return { // Sort by directory level rather than by string length. @@ -870,7 +806,7 @@ function getSource(sourceCode, node) { // Make `../` sort after `../../` but before `../a` etc. // Why a comma? See the next comment. .replace(/^[./]*\/$/, "$&,") - // Make `.` and `/` sort before any other punctation. + // Make `.` and `/` sort before any other punctuation. // The default order is: _ - , x x x . x x x / x x x // We’re changing it to: . / , x x x _ x x x - x x x .replace(/[./_-]/g, (char) => { @@ -889,16 +825,71 @@ function getSource(sourceCode, node) { } }), originalSource: source, - kind: getImportExportKind(node), + kind, }; } +function getSourceTextAndKind(sourceCode, node) { + switch (node.type) { + case "TSImportEqualsDeclaration": + return getSourceTextAndKindFromModuleReference( + sourceCode, + node.moduleReference + ); + default: + return [node.source.value, getImportExportKind(node)]; + } +} + +const KIND_VALUE = "value"; +const KIND_TS_IMPORT_ASSIGNMENT_REQUIRE = "z_require"; +const KIND_TS_IMPORT_ASSIGNMENT_NAMESPACE = "z_namespace"; + +function getSourceTextAndKindFromModuleReference(sourceCode, node) { + switch (node.type) { + case "TSExternalModuleReference": + switch (node.expression.type) { + case "Literal": + return [ + typeof node.expression.value === "string" + ? node.expression.value + : node.expression.raw, + KIND_TS_IMPORT_ASSIGNMENT_REQUIRE, + ]; + default: { + const [start, end] = node.expression.range; + return [ + sourceCode.text.slice(start, end), + KIND_TS_IMPORT_ASSIGNMENT_REQUIRE, + ]; + } + } + case "TSQualifiedName": + return [ + getSourceTextFromTSQualifiedName(sourceCode, node), + KIND_TS_IMPORT_ASSIGNMENT_NAMESPACE, + ]; + default: + return [node.name, KIND_TS_IMPORT_ASSIGNMENT_NAMESPACE]; + } +} + +function getSourceTextFromTSQualifiedName(sourceCode, node) { + switch (node.left.type) { + case "TSQualifiedName": + return `${getSourceTextFromTSQualifiedName(sourceCode, node.left)}.${ + node.right.name + }`; + default: { + return `${node.left.name}.${node.right.name}`; + } + } +} + function getImportExportKind(node) { // `type` and `typeof` imports, as well as `type` exports (there are no - // `typeof` exports). In Flow, import specifiers can also have a kind. Default - // to "value" (like TypeScript) to make regular imports/exports come after the - // type imports/exports. - return node.importKind || node.exportKind || "value"; + // `typeof` exports). + return node.importKind || node.exportKind || KIND_VALUE; } // Like `Array.prototype.findIndex`, but searches from the end. @@ -923,6 +914,9 @@ module.exports = { flatMap, getImportExportItems, isPunctuator, + KIND_TS_IMPORT_ASSIGNMENT_NAMESPACE, + KIND_TS_IMPORT_ASSIGNMENT_REQUIRE, + KIND_VALUE, maybeReportSorting, printSortedItems, printWithSortedSpecifiers, diff --git a/test/__snapshots__/examples.test.js.snap b/test/__snapshots__/examples.test.js.snap index c706ca1..0ff5c65 100644 --- a/test/__snapshots__/examples.test.js.snap +++ b/test/__snapshots__/examples.test.js.snap @@ -496,6 +496,15 @@ import g from "."; import h from "./constants"; import i from "./styles"; +// TypeScript import assignments. +import J = require("../parent"); +import J = require("./sibling"); +export import K = require("an-npm-package"); +import L = require("different-npm-package"); +import M = Namespace; +export import N = Namespace.A.B.C; +import O = Namespace.A.C; + // Different types of exports: export { a } from "../.."; export { b } from "/"; diff --git a/test/imports.test.js b/test/imports.test.js index 417add1..a6ce564 100644 --- a/test/imports.test.js +++ b/test/imports.test.js @@ -1846,6 +1846,13 @@ const typescriptTests = { `import type {} from "a"`, `import type { } from "a"`, `import json from "./foo.json" assert { type: "json" };`, + `import A = B`, + `import A = B.C`, + `import A = require("A")`, + + // These are parsed as type imports, but they are not valid in TypeScript: + `import A = require(1)`, + `import A = require({ b, a })`, // type specifiers. `import { a, type b, c, type d } from "a"`, @@ -2050,6 +2057,7 @@ const typescriptTests = { code: input` |import { Namespace } from './namespace'; |import Foo = Namespace.Foo; + |import old = require('./old'); |import { bar } from './a'; `, output: (actual) => { @@ -2057,6 +2065,7 @@ const typescriptTests = { |import { bar } from './a'; |import { Namespace } from './namespace'; | + |import old = require('./old'); |import Foo = Namespace.Foo; `); }, @@ -2064,48 +2073,88 @@ const typescriptTests = { }, { code: input` + |import _ = require(''); + |import _ = require(""); + |import _ = require(''); |import B = require('./b'); |import A = require('./a'); |import Foo = require('foo'); - |import Bar = require('../foo'); + |import Foo = require('../foo'); + |import Foo = require('../'); + |import Foo = require('..'); + |import At = require("@org/name"); + |import fs = require("node:fs"); `, output: (actual) => { expect(actual).toMatchInlineSnapshot(` - |import Foo = require('foo'); - | - |import Bar = require('../foo'); + |import _ = require(''); + |import _ = require(""); + |import _ = require(''); + |import Foo = require('..'); + |import Foo = require('../'); + |import Foo = require('../foo'); |import A = require('./a'); |import B = require('./b'); + |import At = require("@org/name"); + |import Foo = require('foo'); + |import fs = require("node:fs"); `); }, errors: 1, }, { code: input` - |import Foo = require('foo'); - |import Bar = require(cool().thing); - |import Bar = require(1 + 1); + |import A = require(null); + |import A = require(cool().thing); + |import A = require(1 + 1); + |import A = require({ a }); + |import A = require('foo'); `, output: (actual) => { expect(actual).toMatchInlineSnapshot(` - |import Bar = require(1 + 1); - |import Bar = require(cool().thing); - |import Foo = require('foo'); + |import A = require({ a }); + |import A = require(1 + 1); + |import A = require(cool().thing); + |import A = require('foo'); + |import A = require(null); + `); + }, + errors: 1, + }, + { + code: input` + |/*1*/import/*2*/B/*3*/=/*4*/require/*5*/(/*6*/"B"/*7*/)/*8*/;/*9*//*10 + |*/import B + | //11 + | = require('A'); //12 + `, + output: (actual) => { + expect(actual).toMatchInlineSnapshot(` + |/*10 + |*/import B + | //11 + | = require('A'); //12 + |/*1*/import/*2*/B/*3*/=/*4*/require/*5*/(/*6*/"B"/*7*/)/*8*/;/*9*/ `); }, errors: 1, }, { code: input` - |import B = Namespace/*aaaa*/.B; - |import AB = Namespace.A.B; - |import A = Namespace.A.A; + |/*1*/import/*2*/B/*3*/=/*4*/Namespace/*5*/./*5*/B/*6*/;/*7*//*8 + |*/import AB = Namespace.A.B; //9 + |import A = + | //10 + | Namespace.A.A; `, output: (actual) => { expect(actual).toMatchInlineSnapshot(` - |import A = Namespace.A.A; - |import AB = Namespace.A.B; - |import B = Namespace/*aaaa*/.B; + |import A = + | //10 + | Namespace.A.A; + |/*8 + |*/import AB = Namespace.A.B; //9 + |/*1*/import/*2*/B/*3*/=/*4*/Namespace/*5*/./*5*/B/*6*/;/*7*/ `); }, errors: 1, @@ -2129,6 +2178,74 @@ const typescriptTests = { }, errors: 1, }, + { + options: [{ groups: [] }], + code: input` + |import {} from 'A.B.C'; + |import type {} from 'A.B.C'; + |import A = require('A.B.C'); + |import B = A.B.C; + `, + output: (actual) => { + expect(actual).toMatchInlineSnapshot(` + |import type {} from 'A.B.C'; + |import {} from 'A.B.C'; + |import B = A.B.C; + |import A = require('A.B.C'); + `); + }, + errors: 1, + }, + { + options: [{ groups: [] }], + code: input` + |import type {} from 'D'; + |import {} from 'C'; + |import B = B; + |import A = require('A'); + `, + output: (actual) => { + expect(actual).toMatchInlineSnapshot(` + |import A = require('A'); + |import B = B; + |import {} from 'C'; + |import type {} from 'D'; + `); + }, + errors: 1, + }, + { + options: [{ groups: [] }], + code: input` + |import A = require('./A'); + |import {} from '../parent'; + `, + output: (actual) => { + expect(actual).toMatchInlineSnapshot(` + |import {} from '../parent'; + |import A = require('./A'); + `); + }, + errors: 1, + }, + { + options: [{ groups: [["^\\u0002"], ["^"], ["^\\u0001"]] }], + code: input` + |import A = require('./A'); + |import {} from '../parent'; + |import B = B; + `, + output: (actual) => { + expect(actual).toMatchInlineSnapshot(` + |import B = B; + | + |import {} from '../parent'; + | + |import A = require('./A'); + `); + }, + errors: 1, + }, ], };