463 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			463 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
 | 
						|
 | 
						|
 | 
						|
import {ContextualKeyword} from "../parser/tokenizer/keywords";
 | 
						|
import {TokenType as tt} from "../parser/tokenizer/types";
 | 
						|
 | 
						|
import getClassInfo, {} from "../util/getClassInfo";
 | 
						|
import CJSImportTransformer from "./CJSImportTransformer";
 | 
						|
import ESMImportTransformer from "./ESMImportTransformer";
 | 
						|
import FlowTransformer from "./FlowTransformer";
 | 
						|
import JestHoistTransformer from "./JestHoistTransformer";
 | 
						|
import JSXTransformer from "./JSXTransformer";
 | 
						|
import NumericSeparatorTransformer from "./NumericSeparatorTransformer";
 | 
						|
import OptionalCatchBindingTransformer from "./OptionalCatchBindingTransformer";
 | 
						|
import OptionalChainingNullishTransformer from "./OptionalChainingNullishTransformer";
 | 
						|
import ReactDisplayNameTransformer from "./ReactDisplayNameTransformer";
 | 
						|
import ReactHotLoaderTransformer from "./ReactHotLoaderTransformer";
 | 
						|
 | 
						|
import TypeScriptTransformer from "./TypeScriptTransformer";
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
export default class RootTransformer {
 | 
						|
   __init() {this.transformers = []}
 | 
						|
  
 | 
						|
  
 | 
						|
   __init2() {this.generatedVariables = []}
 | 
						|
  
 | 
						|
  
 | 
						|
  
 | 
						|
  
 | 
						|
 | 
						|
  constructor(
 | 
						|
    sucraseContext,
 | 
						|
    transforms,
 | 
						|
    enableLegacyBabel5ModuleInterop,
 | 
						|
    options,
 | 
						|
  ) {;RootTransformer.prototype.__init.call(this);RootTransformer.prototype.__init2.call(this);
 | 
						|
    this.nameManager = sucraseContext.nameManager;
 | 
						|
    this.helperManager = sucraseContext.helperManager;
 | 
						|
    const {tokenProcessor, importProcessor} = sucraseContext;
 | 
						|
    this.tokens = tokenProcessor;
 | 
						|
    this.isImportsTransformEnabled = transforms.includes("imports");
 | 
						|
    this.isReactHotLoaderTransformEnabled = transforms.includes("react-hot-loader");
 | 
						|
    this.disableESTransforms = Boolean(options.disableESTransforms);
 | 
						|
 | 
						|
    if (!options.disableESTransforms) {
 | 
						|
      this.transformers.push(
 | 
						|
        new OptionalChainingNullishTransformer(tokenProcessor, this.nameManager),
 | 
						|
      );
 | 
						|
      this.transformers.push(new NumericSeparatorTransformer(tokenProcessor));
 | 
						|
      this.transformers.push(new OptionalCatchBindingTransformer(tokenProcessor, this.nameManager));
 | 
						|
    }
 | 
						|
 | 
						|
    if (transforms.includes("jsx")) {
 | 
						|
      if (options.jsxRuntime !== "preserve") {
 | 
						|
        this.transformers.push(
 | 
						|
          new JSXTransformer(this, tokenProcessor, importProcessor, this.nameManager, options),
 | 
						|
        );
 | 
						|
      }
 | 
						|
      this.transformers.push(
 | 
						|
        new ReactDisplayNameTransformer(this, tokenProcessor, importProcessor, options),
 | 
						|
      );
 | 
						|
    }
 | 
						|
 | 
						|
    let reactHotLoaderTransformer = null;
 | 
						|
    if (transforms.includes("react-hot-loader")) {
 | 
						|
      if (!options.filePath) {
 | 
						|
        throw new Error("filePath is required when using the react-hot-loader transform.");
 | 
						|
      }
 | 
						|
      reactHotLoaderTransformer = new ReactHotLoaderTransformer(tokenProcessor, options.filePath);
 | 
						|
      this.transformers.push(reactHotLoaderTransformer);
 | 
						|
    }
 | 
						|
 | 
						|
    // Note that we always want to enable the imports transformer, even when the import transform
 | 
						|
    // itself isn't enabled, since we need to do type-only import pruning for both Flow and
 | 
						|
    // TypeScript.
 | 
						|
    if (transforms.includes("imports")) {
 | 
						|
      if (importProcessor === null) {
 | 
						|
        throw new Error("Expected non-null importProcessor with imports transform enabled.");
 | 
						|
      }
 | 
						|
      this.transformers.push(
 | 
						|
        new CJSImportTransformer(
 | 
						|
          this,
 | 
						|
          tokenProcessor,
 | 
						|
          importProcessor,
 | 
						|
          this.nameManager,
 | 
						|
          this.helperManager,
 | 
						|
          reactHotLoaderTransformer,
 | 
						|
          enableLegacyBabel5ModuleInterop,
 | 
						|
          Boolean(options.enableLegacyTypeScriptModuleInterop),
 | 
						|
          transforms.includes("typescript"),
 | 
						|
          transforms.includes("flow"),
 | 
						|
          Boolean(options.preserveDynamicImport),
 | 
						|
          Boolean(options.keepUnusedImports),
 | 
						|
        ),
 | 
						|
      );
 | 
						|
    } else {
 | 
						|
      this.transformers.push(
 | 
						|
        new ESMImportTransformer(
 | 
						|
          tokenProcessor,
 | 
						|
          this.nameManager,
 | 
						|
          this.helperManager,
 | 
						|
          reactHotLoaderTransformer,
 | 
						|
          transforms.includes("typescript"),
 | 
						|
          transforms.includes("flow"),
 | 
						|
          Boolean(options.keepUnusedImports),
 | 
						|
          options,
 | 
						|
        ),
 | 
						|
      );
 | 
						|
    }
 | 
						|
 | 
						|
    if (transforms.includes("flow")) {
 | 
						|
      this.transformers.push(
 | 
						|
        new FlowTransformer(this, tokenProcessor, transforms.includes("imports")),
 | 
						|
      );
 | 
						|
    }
 | 
						|
    if (transforms.includes("typescript")) {
 | 
						|
      this.transformers.push(
 | 
						|
        new TypeScriptTransformer(this, tokenProcessor, transforms.includes("imports")),
 | 
						|
      );
 | 
						|
    }
 | 
						|
    if (transforms.includes("jest")) {
 | 
						|
      this.transformers.push(
 | 
						|
        new JestHoistTransformer(this, tokenProcessor, this.nameManager, importProcessor),
 | 
						|
      );
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  transform() {
 | 
						|
    this.tokens.reset();
 | 
						|
    this.processBalancedCode();
 | 
						|
    const shouldAddUseStrict = this.isImportsTransformEnabled;
 | 
						|
    // "use strict" always needs to be first, so override the normal transformer order.
 | 
						|
    let prefix = shouldAddUseStrict ? '"use strict";' : "";
 | 
						|
    for (const transformer of this.transformers) {
 | 
						|
      prefix += transformer.getPrefixCode();
 | 
						|
    }
 | 
						|
    prefix += this.helperManager.emitHelpers();
 | 
						|
    prefix += this.generatedVariables.map((v) => ` var ${v};`).join("");
 | 
						|
    for (const transformer of this.transformers) {
 | 
						|
      prefix += transformer.getHoistedCode();
 | 
						|
    }
 | 
						|
    let suffix = "";
 | 
						|
    for (const transformer of this.transformers) {
 | 
						|
      suffix += transformer.getSuffixCode();
 | 
						|
    }
 | 
						|
    const result = this.tokens.finish();
 | 
						|
    let {code} = result;
 | 
						|
    if (code.startsWith("#!")) {
 | 
						|
      let newlineIndex = code.indexOf("\n");
 | 
						|
      if (newlineIndex === -1) {
 | 
						|
        newlineIndex = code.length;
 | 
						|
        code += "\n";
 | 
						|
      }
 | 
						|
      return {
 | 
						|
        code: code.slice(0, newlineIndex + 1) + prefix + code.slice(newlineIndex + 1) + suffix,
 | 
						|
        // The hashbang line has no tokens, so shifting the tokens to account
 | 
						|
        // for prefix can happen normally.
 | 
						|
        mappings: this.shiftMappings(result.mappings, prefix.length),
 | 
						|
      };
 | 
						|
    } else {
 | 
						|
      return {
 | 
						|
        code: prefix + code + suffix,
 | 
						|
        mappings: this.shiftMappings(result.mappings, prefix.length),
 | 
						|
      };
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  processBalancedCode() {
 | 
						|
    let braceDepth = 0;
 | 
						|
    let parenDepth = 0;
 | 
						|
    while (!this.tokens.isAtEnd()) {
 | 
						|
      if (this.tokens.matches1(tt.braceL) || this.tokens.matches1(tt.dollarBraceL)) {
 | 
						|
        braceDepth++;
 | 
						|
      } else if (this.tokens.matches1(tt.braceR)) {
 | 
						|
        if (braceDepth === 0) {
 | 
						|
          return;
 | 
						|
        }
 | 
						|
        braceDepth--;
 | 
						|
      }
 | 
						|
      if (this.tokens.matches1(tt.parenL)) {
 | 
						|
        parenDepth++;
 | 
						|
      } else if (this.tokens.matches1(tt.parenR)) {
 | 
						|
        if (parenDepth === 0) {
 | 
						|
          return;
 | 
						|
        }
 | 
						|
        parenDepth--;
 | 
						|
      }
 | 
						|
      this.processToken();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  processToken() {
 | 
						|
    if (this.tokens.matches1(tt._class)) {
 | 
						|
      this.processClass();
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    for (const transformer of this.transformers) {
 | 
						|
      const wasProcessed = transformer.process();
 | 
						|
      if (wasProcessed) {
 | 
						|
        return;
 | 
						|
      }
 | 
						|
    }
 | 
						|
    this.tokens.copyToken();
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Skip past a class with a name and return that name.
 | 
						|
   */
 | 
						|
  processNamedClass() {
 | 
						|
    if (!this.tokens.matches2(tt._class, tt.name)) {
 | 
						|
      throw new Error("Expected identifier for exported class name.");
 | 
						|
    }
 | 
						|
    const name = this.tokens.identifierNameAtIndex(this.tokens.currentIndex() + 1);
 | 
						|
    this.processClass();
 | 
						|
    return name;
 | 
						|
  }
 | 
						|
 | 
						|
  processClass() {
 | 
						|
    const classInfo = getClassInfo(this, this.tokens, this.nameManager, this.disableESTransforms);
 | 
						|
 | 
						|
    // Both static and instance initializers need a class name to use to invoke the initializer, so
 | 
						|
    // assign to one if necessary.
 | 
						|
    const needsCommaExpression =
 | 
						|
      (classInfo.headerInfo.isExpression || !classInfo.headerInfo.className) &&
 | 
						|
      classInfo.staticInitializerNames.length + classInfo.instanceInitializerNames.length > 0;
 | 
						|
 | 
						|
    let className = classInfo.headerInfo.className;
 | 
						|
    if (needsCommaExpression) {
 | 
						|
      className = this.nameManager.claimFreeName("_class");
 | 
						|
      this.generatedVariables.push(className);
 | 
						|
      this.tokens.appendCode(` (${className} =`);
 | 
						|
    }
 | 
						|
 | 
						|
    const classToken = this.tokens.currentToken();
 | 
						|
    const contextId = classToken.contextId;
 | 
						|
    if (contextId == null) {
 | 
						|
      throw new Error("Expected class to have a context ID.");
 | 
						|
    }
 | 
						|
    this.tokens.copyExpectedToken(tt._class);
 | 
						|
    while (!this.tokens.matchesContextIdAndLabel(tt.braceL, contextId)) {
 | 
						|
      this.processToken();
 | 
						|
    }
 | 
						|
 | 
						|
    this.processClassBody(classInfo, className);
 | 
						|
 | 
						|
    const staticInitializerStatements = classInfo.staticInitializerNames.map(
 | 
						|
      (name) => `${className}.${name}()`,
 | 
						|
    );
 | 
						|
    if (needsCommaExpression) {
 | 
						|
      this.tokens.appendCode(
 | 
						|
        `, ${staticInitializerStatements.map((s) => `${s}, `).join("")}${className})`,
 | 
						|
      );
 | 
						|
    } else if (classInfo.staticInitializerNames.length > 0) {
 | 
						|
      this.tokens.appendCode(` ${staticInitializerStatements.map((s) => `${s};`).join(" ")}`);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * We want to just handle class fields in all contexts, since TypeScript supports them. Later,
 | 
						|
   * when some JS implementations support class fields, this should be made optional.
 | 
						|
   */
 | 
						|
  processClassBody(classInfo, className) {
 | 
						|
    const {
 | 
						|
      headerInfo,
 | 
						|
      constructorInsertPos,
 | 
						|
      constructorInitializerStatements,
 | 
						|
      fields,
 | 
						|
      instanceInitializerNames,
 | 
						|
      rangesToRemove,
 | 
						|
    } = classInfo;
 | 
						|
    let fieldIndex = 0;
 | 
						|
    let rangeToRemoveIndex = 0;
 | 
						|
    const classContextId = this.tokens.currentToken().contextId;
 | 
						|
    if (classContextId == null) {
 | 
						|
      throw new Error("Expected non-null context ID on class.");
 | 
						|
    }
 | 
						|
    this.tokens.copyExpectedToken(tt.braceL);
 | 
						|
    if (this.isReactHotLoaderTransformEnabled) {
 | 
						|
      this.tokens.appendCode(
 | 
						|
        "__reactstandin__regenerateByEval(key, code) {this[key] = eval(code);}",
 | 
						|
      );
 | 
						|
    }
 | 
						|
 | 
						|
    const needsConstructorInit =
 | 
						|
      constructorInitializerStatements.length + instanceInitializerNames.length > 0;
 | 
						|
 | 
						|
    if (constructorInsertPos === null && needsConstructorInit) {
 | 
						|
      const constructorInitializersCode = this.makeConstructorInitCode(
 | 
						|
        constructorInitializerStatements,
 | 
						|
        instanceInitializerNames,
 | 
						|
        className,
 | 
						|
      );
 | 
						|
      if (headerInfo.hasSuperclass) {
 | 
						|
        const argsName = this.nameManager.claimFreeName("args");
 | 
						|
        this.tokens.appendCode(
 | 
						|
          `constructor(...${argsName}) { super(...${argsName}); ${constructorInitializersCode}; }`,
 | 
						|
        );
 | 
						|
      } else {
 | 
						|
        this.tokens.appendCode(`constructor() { ${constructorInitializersCode}; }`);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    while (!this.tokens.matchesContextIdAndLabel(tt.braceR, classContextId)) {
 | 
						|
      if (fieldIndex < fields.length && this.tokens.currentIndex() === fields[fieldIndex].start) {
 | 
						|
        let needsCloseBrace = false;
 | 
						|
        if (this.tokens.matches1(tt.bracketL)) {
 | 
						|
          this.tokens.copyTokenWithPrefix(`${fields[fieldIndex].initializerName}() {this`);
 | 
						|
        } else if (this.tokens.matches1(tt.string) || this.tokens.matches1(tt.num)) {
 | 
						|
          this.tokens.copyTokenWithPrefix(`${fields[fieldIndex].initializerName}() {this[`);
 | 
						|
          needsCloseBrace = true;
 | 
						|
        } else {
 | 
						|
          this.tokens.copyTokenWithPrefix(`${fields[fieldIndex].initializerName}() {this.`);
 | 
						|
        }
 | 
						|
        while (this.tokens.currentIndex() < fields[fieldIndex].end) {
 | 
						|
          if (needsCloseBrace && this.tokens.currentIndex() === fields[fieldIndex].equalsIndex) {
 | 
						|
            this.tokens.appendCode("]");
 | 
						|
          }
 | 
						|
          this.processToken();
 | 
						|
        }
 | 
						|
        this.tokens.appendCode("}");
 | 
						|
        fieldIndex++;
 | 
						|
      } else if (
 | 
						|
        rangeToRemoveIndex < rangesToRemove.length &&
 | 
						|
        this.tokens.currentIndex() >= rangesToRemove[rangeToRemoveIndex].start
 | 
						|
      ) {
 | 
						|
        if (this.tokens.currentIndex() < rangesToRemove[rangeToRemoveIndex].end) {
 | 
						|
          this.tokens.removeInitialToken();
 | 
						|
        }
 | 
						|
        while (this.tokens.currentIndex() < rangesToRemove[rangeToRemoveIndex].end) {
 | 
						|
          this.tokens.removeToken();
 | 
						|
        }
 | 
						|
        rangeToRemoveIndex++;
 | 
						|
      } else if (this.tokens.currentIndex() === constructorInsertPos) {
 | 
						|
        this.tokens.copyToken();
 | 
						|
        if (needsConstructorInit) {
 | 
						|
          this.tokens.appendCode(
 | 
						|
            `;${this.makeConstructorInitCode(
 | 
						|
              constructorInitializerStatements,
 | 
						|
              instanceInitializerNames,
 | 
						|
              className,
 | 
						|
            )};`,
 | 
						|
          );
 | 
						|
        }
 | 
						|
        this.processToken();
 | 
						|
      } else {
 | 
						|
        this.processToken();
 | 
						|
      }
 | 
						|
    }
 | 
						|
    this.tokens.copyExpectedToken(tt.braceR);
 | 
						|
  }
 | 
						|
 | 
						|
  makeConstructorInitCode(
 | 
						|
    constructorInitializerStatements,
 | 
						|
    instanceInitializerNames,
 | 
						|
    className,
 | 
						|
  ) {
 | 
						|
    return [
 | 
						|
      ...constructorInitializerStatements,
 | 
						|
      ...instanceInitializerNames.map((name) => `${className}.prototype.${name}.call(this)`),
 | 
						|
    ].join(";");
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Normally it's ok to simply remove type tokens, but we need to be more careful when dealing with
 | 
						|
   * arrow function return types since they can confuse the parser. In that case, we want to move
 | 
						|
   * the close-paren to the same line as the arrow.
 | 
						|
   *
 | 
						|
   * See https://github.com/alangpierce/sucrase/issues/391 for more details.
 | 
						|
   */
 | 
						|
  processPossibleArrowParamEnd() {
 | 
						|
    if (this.tokens.matches2(tt.parenR, tt.colon) && this.tokens.tokenAtRelativeIndex(1).isType) {
 | 
						|
      let nextNonTypeIndex = this.tokens.currentIndex() + 1;
 | 
						|
      // Look ahead to see if this is an arrow function or something else.
 | 
						|
      while (this.tokens.tokens[nextNonTypeIndex].isType) {
 | 
						|
        nextNonTypeIndex++;
 | 
						|
      }
 | 
						|
      if (this.tokens.matches1AtIndex(nextNonTypeIndex, tt.arrow)) {
 | 
						|
        this.tokens.removeInitialToken();
 | 
						|
        while (this.tokens.currentIndex() < nextNonTypeIndex) {
 | 
						|
          this.tokens.removeToken();
 | 
						|
        }
 | 
						|
        this.tokens.replaceTokenTrimmingLeftWhitespace(") =>");
 | 
						|
        return true;
 | 
						|
      }
 | 
						|
    }
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * An async arrow function might be of the form:
 | 
						|
   *
 | 
						|
   * async <
 | 
						|
   *   T
 | 
						|
   * >() => {}
 | 
						|
   *
 | 
						|
   * in which case, removing the type parameters will cause a syntax error. Detect this case and
 | 
						|
   * move the open-paren earlier.
 | 
						|
   */
 | 
						|
  processPossibleAsyncArrowWithTypeParams() {
 | 
						|
    if (
 | 
						|
      !this.tokens.matchesContextual(ContextualKeyword._async) &&
 | 
						|
      !this.tokens.matches1(tt._async)
 | 
						|
    ) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
    const nextToken = this.tokens.tokenAtRelativeIndex(1);
 | 
						|
    if (nextToken.type !== tt.lessThan || !nextToken.isType) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
 | 
						|
    let nextNonTypeIndex = this.tokens.currentIndex() + 1;
 | 
						|
    // Look ahead to see if this is an arrow function or something else.
 | 
						|
    while (this.tokens.tokens[nextNonTypeIndex].isType) {
 | 
						|
      nextNonTypeIndex++;
 | 
						|
    }
 | 
						|
    if (this.tokens.matches1AtIndex(nextNonTypeIndex, tt.parenL)) {
 | 
						|
      this.tokens.replaceToken("async (");
 | 
						|
      this.tokens.removeInitialToken();
 | 
						|
      while (this.tokens.currentIndex() < nextNonTypeIndex) {
 | 
						|
        this.tokens.removeToken();
 | 
						|
      }
 | 
						|
      this.tokens.removeToken();
 | 
						|
      // We ate a ( token, so we need to process the tokens in between and then the ) token so that
 | 
						|
      // we remain balanced.
 | 
						|
      this.processBalancedCode();
 | 
						|
      this.processToken();
 | 
						|
      return true;
 | 
						|
    }
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  processPossibleTypeRange() {
 | 
						|
    if (this.tokens.currentToken().isType) {
 | 
						|
      this.tokens.removeInitialToken();
 | 
						|
      while (this.tokens.currentToken().isType) {
 | 
						|
        this.tokens.removeToken();
 | 
						|
      }
 | 
						|
      return true;
 | 
						|
    }
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  shiftMappings(
 | 
						|
    mappings,
 | 
						|
    prefixLength,
 | 
						|
  ) {
 | 
						|
    for (let i = 0; i < mappings.length; i++) {
 | 
						|
      const mapping = mappings[i];
 | 
						|
      if (mapping !== undefined) {
 | 
						|
        mappings[i] = mapping + prefixLength;
 | 
						|
      }
 | 
						|
    }
 | 
						|
    return mappings;
 | 
						|
  }
 | 
						|
}
 |