From 57323b6b13dce9cb47ddc87a159f713b0a025d74 Mon Sep 17 00:00:00 2001 From: Alan Pierce Date: Sun, 18 Nov 2018 17:10:45 -0800 Subject: [PATCH] Properly elide TS imports even when referenced by shadowing variables (#342) Fixes #298 TypeScript is required to remove imports where all bindings are never referenced in a value position, but previously it could have false positives when there are shadowed variables with the same name. Now, we do the intelligent scope-based shadowed global detection like we do with import replacement and properly elide when necessary. --- src/index.ts | 7 ++- src/util/getNonTypeIdentifiers.ts | 3 +- src/util/getTSImportedNames.ts | 88 +++++++++++++++++++++++++++++++ test/typescript-test.ts | 54 +++++++++++++++++++ 4 files changed, 150 insertions(+), 2 deletions(-) create mode 100644 src/util/getTSImportedNames.ts diff --git a/src/index.ts b/src/index.ts index 805d3c0a..6e70b842 100644 --- a/src/index.ts +++ b/src/index.ts @@ -7,6 +7,7 @@ import {Scope} from "./parser/tokenizer/state"; import TokenProcessor from "./TokenProcessor"; import RootTransformer from "./transformers/RootTransformer"; import formatTokens from "./util/formatTokens"; +import getTSImportedNames from "./util/getTSImportedNames"; export type Transform = "jsx" | "typescript" | "flow" | "imports"; @@ -136,10 +137,14 @@ function getSucraseContext(code: string, options: Options): SucraseContext { enableLegacyTypeScriptModuleInterop, ); importProcessor.preprocessTokens(); + // We need to mark shadowed globals after processing imports so we know that the globals are, + // but before type-only import pruning, since that relies on shadowing information. + identifyShadowedGlobals(tokenProcessor, scopes, importProcessor.getGlobalNames()); if (options.transforms.includes("typescript")) { importProcessor.pruneTypeOnlyImports(); } - identifyShadowedGlobals(tokenProcessor, scopes, importProcessor.getGlobalNames()); + } else if (options.transforms.includes("typescript")) { + identifyShadowedGlobals(tokenProcessor, scopes, getTSImportedNames(tokenProcessor)); } return {tokenProcessor, scopes, nameManager, importProcessor}; } diff --git a/src/util/getNonTypeIdentifiers.ts b/src/util/getNonTypeIdentifiers.ts index 4d8a1bc1..095fc09e 100644 --- a/src/util/getNonTypeIdentifiers.ts +++ b/src/util/getNonTypeIdentifiers.ts @@ -12,7 +12,8 @@ export function getNonTypeIdentifiers(tokens: TokenProcessor): Set { !token.isType && (token.identifierRole === IdentifierRole.Access || token.identifierRole === IdentifierRole.ObjectShorthand || - token.identifierRole === IdentifierRole.ExportAccess) + token.identifierRole === IdentifierRole.ExportAccess) && + !token.shadowsGlobal ) { nonTypeIdentifiers.add(tokens.identifierNameForToken(token)); } diff --git a/src/util/getTSImportedNames.ts b/src/util/getTSImportedNames.ts new file mode 100644 index 00000000..0d9e50a8 --- /dev/null +++ b/src/util/getTSImportedNames.ts @@ -0,0 +1,88 @@ +import {ContextualKeyword} from "../parser/tokenizer"; +import {TokenType as tt} from "../parser/tokenizer/types"; +import TokenProcessor from "../TokenProcessor"; + +/** + * Special case code to scan for imported names in ESM TypeScript. We need to do this so we can + * properly get globals so we can compute shadowed globals. + * + * This is similar to logic in CJSImportProcessor, but trimmed down to avoid logic with CJS + * replacement and flow type imports. + */ +export default function getTSImportedNames(tokens: TokenProcessor): Set { + const importedNames = new Set(); + for (let i = 0; i < tokens.tokens.length; i++) { + if ( + tokens.matchesAtIndex(i, [tt._import]) && + !tokens.matchesAtIndex(i, [tt._import, tt.name, tt.eq]) + ) { + collectNamesForImport(tokens, i, importedNames); + } + } + return importedNames; +} + +function collectNamesForImport( + tokens: TokenProcessor, + index: number, + importedNames: Set, +): void { + index++; + + if (tokens.matchesAtIndex(index, [tt.parenL])) { + // Dynamic import, so nothing to do + return; + } + + if (tokens.matchesAtIndex(index, [tt.name])) { + importedNames.add(tokens.identifierNameAtIndex(index)); + index++; + if (tokens.matchesAtIndex(index, [tt.comma])) { + index++; + } + } + + if (tokens.matchesAtIndex(index, [tt.star])) { + // * as + index += 2; + importedNames.add(tokens.identifierNameAtIndex(index)); + index++; + } + + if (tokens.matchesAtIndex(index, [tt.braceL])) { + index++; + collectNamesForNamedImport(tokens, index, importedNames); + } +} + +function collectNamesForNamedImport( + tokens: TokenProcessor, + index: number, + importedNames: Set, +): void { + while (true) { + if (tokens.matchesAtIndex(index, [tt.braceR])) { + return; + } + + // We care about the local name, which might be the first token, or if there's an "as", is the + // one after that. + let name = tokens.identifierNameAtIndex(index); + index++; + if (tokens.matchesContextualAtIndex(index, ContextualKeyword._as)) { + index++; + name = tokens.identifierNameAtIndex(index); + index++; + } + importedNames.add(name); + if (tokens.matchesAtIndex(index, [tt.comma, tt.braceR])) { + return; + } else if (tokens.matchesAtIndex(index, [tt.braceR])) { + return; + } else if (tokens.matchesAtIndex(index, [tt.comma])) { + index++; + } else { + throw new Error(`Unexpected token: ${JSON.stringify(tokens.tokens[index])}`); + } + } +} diff --git a/test/typescript-test.ts b/test/typescript-test.ts index 5b260a19..acda442c 100644 --- a/test/typescript-test.ts +++ b/test/typescript-test.ts @@ -1270,4 +1270,58 @@ describe("typescript transform", () => { `, ); }); + + it("properly elides CJS imports that only have value references in shadowed names", () => { + assertTypeScriptResult( + ` + import T from './T'; + + const x: T = 3; + + function foo() { + let T = 3; + console.log(T); + } + `, + `"use strict";${IMPORT_DEFAULT_PREFIX} + + + const x = 3; + + function foo() { + let T = 3; + console.log(T); + } + `, + ); + }); + + it("properly elides ESM imports that only have value references in shadowed names", () => { + assertTypeScriptESMResult( + ` + import T, {a as b, c} from './T'; + import {d, e} from './foo'; + + const x: T = 3; + console.log(e); + + function foo() { + let T = 3, b = 4, c = 5, d = 6; + console.log(T, b, c, d); + } + `, + ` + + import { e} from './foo'; + + const x = 3; + console.log(e); + + function foo() { + let T = 3, b = 4, c = 5, d = 6; + console.log(T, b, c, d); + } + `, + ); + }); });