babel-plugin-ember-template-compilation
Advanced tools
Comparing version 1.1.0-alpha.1 to 2.0.0-alpha.0
{ | ||
"name": "babel-plugin-ember-template-compilation", | ||
"version": "1.1.0-alpha.1", | ||
"version": "2.0.0-alpha.0", | ||
"description": "Babel implementation of Ember's low-level template-compilation API", | ||
@@ -47,2 +47,3 @@ "repository": "https://github.com/emberjs/babel-plugin-ember-template-compilation", | ||
"@types/jest": "^26.0.23", | ||
"@types/sinon": "^10.0.13", | ||
"@typescript-eslint/eslint-plugin": "^4.28.4", | ||
@@ -60,2 +61,3 @@ "@typescript-eslint/parser": "^4.28.4", | ||
"release-it-lerna-changelog": "^3.1.0", | ||
"sinon": "^14.0.0", | ||
"typescript": "^4.3.5" | ||
@@ -62,0 +64,0 @@ }, |
import type { types as t } from '@babel/core'; | ||
import type * as Babel from '@babel/core'; | ||
import type { NodePath } from '@babel/traverse'; | ||
import type { ASTv1, WalkerPath } from '@glimmer/syntax'; | ||
import type { ASTPluginBuilder, ASTPluginEnvironment, ASTv1, WalkerPath } from '@glimmer/syntax'; | ||
import type { ImportUtil } from 'babel-import-util'; | ||
import type { State } from './plugin'; | ||
export declare class JSUtils { | ||
#private; | ||
constructor(babel: typeof Babel, program: NodePath<t.Program>, template: NodePath<t.Expression>, locals: string[], importer: ImportUtil); | ||
constructor(babel: typeof Babel, state: State<unknown>, template: NodePath<t.Expression>, locals: string[], importer: ImportUtil); | ||
/** | ||
@@ -70,2 +71,3 @@ * Create a new binding that you can use in your template, initialized with | ||
} & T; | ||
export declare type ExtendedPluginBuilder = ASTPluginBuilder<WithJSUtils<ASTPluginEnvironment>>; | ||
/** | ||
@@ -72,0 +74,0 @@ * Allows you to construct an expression that relies on imported values. |
@@ -13,3 +13,3 @@ "use strict"; | ||
}; | ||
var _JSUtils_instances, _JSUtils_babel, _JSUtils_program, _JSUtils_template, _JSUtils_locals, _JSUtils_importer, _JSUtils_lastInsertedPath, _JSUtils_emitStatement, _JSUtils_parseExpression, _ExpressionContext_importer, _ExpressionContext_target; | ||
var _JSUtils_instances, _JSUtils_babel, _JSUtils_state, _JSUtils_template, _JSUtils_locals, _JSUtils_importer, _JSUtils_emitStatement, _JSUtils_parseExpression, _ExpressionContext_importer, _ExpressionContext_target; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
@@ -20,12 +20,11 @@ exports.JSUtils = void 0; | ||
class JSUtils { | ||
constructor(babel, program, template, locals, importer) { | ||
constructor(babel, state, template, locals, importer) { | ||
_JSUtils_instances.add(this); | ||
_JSUtils_babel.set(this, void 0); | ||
_JSUtils_program.set(this, void 0); | ||
_JSUtils_state.set(this, void 0); | ||
_JSUtils_template.set(this, void 0); | ||
_JSUtils_locals.set(this, void 0); | ||
_JSUtils_importer.set(this, void 0); | ||
_JSUtils_lastInsertedPath.set(this, void 0); | ||
__classPrivateFieldSet(this, _JSUtils_babel, babel, "f"); | ||
__classPrivateFieldSet(this, _JSUtils_program, program, "f"); | ||
__classPrivateFieldSet(this, _JSUtils_state, state, "f"); | ||
__classPrivateFieldSet(this, _JSUtils_template, template, "f"); | ||
@@ -53,5 +52,6 @@ __classPrivateFieldSet(this, _JSUtils_locals, locals, "f"); | ||
let t = __classPrivateFieldGet(this, _JSUtils_babel, "f").types; | ||
__classPrivateFieldGet(this, _JSUtils_instances, "m", _JSUtils_emitStatement).call(this, t.variableDeclaration('let', [ | ||
t.variableDeclarator(t.identifier(name), __classPrivateFieldGet(this, _JSUtils_instances, "m", _JSUtils_parseExpression).call(this, __classPrivateFieldGet(this, _JSUtils_program, "f"), expression)), | ||
let declaration = __classPrivateFieldGet(this, _JSUtils_instances, "m", _JSUtils_emitStatement).call(this, t.variableDeclaration('let', [ | ||
t.variableDeclarator(t.identifier(name), __classPrivateFieldGet(this, _JSUtils_instances, "m", _JSUtils_parseExpression).call(this, __classPrivateFieldGet(this, _JSUtils_state, "f").program, expression)), | ||
])); | ||
declaration.scope.registerBinding('module', declaration.get('declarations.0')); | ||
__classPrivateFieldGet(this, _JSUtils_locals, "f").push(name); | ||
@@ -113,13 +113,14 @@ return name; | ||
let t = __classPrivateFieldGet(this, _JSUtils_babel, "f").types; | ||
__classPrivateFieldGet(this, _JSUtils_instances, "m", _JSUtils_emitStatement).call(this, t.expressionStatement(__classPrivateFieldGet(this, _JSUtils_instances, "m", _JSUtils_parseExpression).call(this, __classPrivateFieldGet(this, _JSUtils_program, "f"), expression))); | ||
__classPrivateFieldGet(this, _JSUtils_instances, "m", _JSUtils_emitStatement).call(this, t.expressionStatement(__classPrivateFieldGet(this, _JSUtils_instances, "m", _JSUtils_parseExpression).call(this, __classPrivateFieldGet(this, _JSUtils_state, "f").program, expression))); | ||
} | ||
} | ||
exports.JSUtils = JSUtils; | ||
_JSUtils_babel = new WeakMap(), _JSUtils_program = new WeakMap(), _JSUtils_template = new WeakMap(), _JSUtils_locals = new WeakMap(), _JSUtils_importer = new WeakMap(), _JSUtils_lastInsertedPath = new WeakMap(), _JSUtils_instances = new WeakSet(), _JSUtils_emitStatement = function _JSUtils_emitStatement(statement) { | ||
if (__classPrivateFieldGet(this, _JSUtils_lastInsertedPath, "f")) { | ||
__classPrivateFieldGet(this, _JSUtils_lastInsertedPath, "f").insertAfter(statement); | ||
_JSUtils_babel = new WeakMap(), _JSUtils_state = new WeakMap(), _JSUtils_template = new WeakMap(), _JSUtils_locals = new WeakMap(), _JSUtils_importer = new WeakMap(), _JSUtils_instances = new WeakSet(), _JSUtils_emitStatement = function _JSUtils_emitStatement(statement) { | ||
if (__classPrivateFieldGet(this, _JSUtils_state, "f").lastInsertedPath) { | ||
__classPrivateFieldGet(this, _JSUtils_state, "f").lastInsertedPath = __classPrivateFieldGet(this, _JSUtils_state, "f").lastInsertedPath.insertAfter(statement)[0]; | ||
} | ||
else { | ||
__classPrivateFieldSet(this, _JSUtils_lastInsertedPath, __classPrivateFieldGet(this, _JSUtils_program, "f").unshiftContainer('body', statement)[0], "f"); | ||
__classPrivateFieldGet(this, _JSUtils_state, "f").lastInsertedPath = __classPrivateFieldGet(this, _JSUtils_state, "f").program.unshiftContainer('body', statement)[0]; | ||
} | ||
return __classPrivateFieldGet(this, _JSUtils_state, "f").lastInsertedPath; | ||
}, _JSUtils_parseExpression = function _JSUtils_parseExpression(target, expression) { | ||
@@ -211,2 +212,2 @@ let expressionString; | ||
_ExpressionContext_importer = new WeakMap(), _ExpressionContext_target = new WeakMap(); | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"js-utils.js","sourceRoot":"","sources":["js-utils.ts"],"names":[],"mappings":"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","sourcesContent":["import type { types as t } from '@babel/core';\nimport type * as Babel from '@babel/core';\nimport type { NodePath } from '@babel/traverse';\nimport type { ASTv1, WalkerPath } from '@glimmer/syntax';\nimport type { ImportUtil } from 'babel-import-util';\n\n// This exists to give AST plugins a controlled interface for influencing the\n// surrounding Javascript scope\nexport class JSUtils {\n  #babel: typeof Babel;\n  #program: NodePath<t.Program>;\n  #template: NodePath<t.Expression>;\n  #locals: string[];\n  #importer: ImportUtil;\n  #lastInsertedPath: NodePath<t.Node> | undefined;\n\n  constructor(\n    babel: typeof Babel,\n    program: NodePath<t.Program>,\n    template: NodePath<t.Expression>,\n    locals: string[],\n    importer: ImportUtil\n  ) {\n    this.#babel = babel;\n    this.#program = program;\n    this.#template = template;\n    this.#locals = locals;\n    this.#importer = importer;\n  }\n\n  /**\n   * Create a new binding that you can use in your template, initialized with\n   * the given Javascript expression.\n   *\n   * @param { Expression } expression A javascript expression whose value will\n   * initialize your new binding. See docs on the Expression type for details.\n   * @param target The location within your template where the binding will be\n   * used. This matters so we can avoid naming collisions.\n   * @param opts.nameHint Optionally, provide a descriptive name for your new\n   * binding. We will mangle this name as needed to avoid collisions, but\n   * picking a good name here can aid in debugging.\n   *\n   * @return The name you can use in your template to access the binding.\n   */\n  bindExpression(\n    expression: Expression,\n    target: WalkerPath<ASTv1.Node>,\n    opts?: { nameHint?: string }\n  ): string {\n    let name = unusedNameLike(\n      opts?.nameHint ?? 'a',\n      (candidate) =>\n        this.#template.scope.hasBinding(candidate) || astNodeHasBinding(target, candidate)\n    );\n    let t = this.#babel.types;\n    this.#emitStatement(\n      t.variableDeclaration('let', [\n        t.variableDeclarator(t.identifier(name), this.#parseExpression(this.#program, expression)),\n      ])\n    );\n    this.#locals.push(name);\n    return name;\n  }\n\n  #emitStatement(statement: t.Statement): void {\n    if (this.#lastInsertedPath) {\n      this.#lastInsertedPath.insertAfter(statement);\n    } else {\n      this.#lastInsertedPath = this.#program.unshiftContainer('body', statement)[0];\n    }\n  }\n\n  /**\n   * Gain access to an imported value within your template.\n   *\n   * @param moduleSpecifier The path to import from.\n   * @param exportedName The named export you wish to access, or \"default\" for\n   * the default export, or \"*\" for the namespace export.\n   * @param target The location within your template where the binding will be\n   * used. This matters so we can avoid naming collisions.\n   * @param opts.nameHint Optionally, provide a descriptive name for your new\n   * binding. We will mangle this name as needed to avoid collisions, but\n   * picking a good name here can aid in debugging.\n   *\n   * @return The name you can use in your template to access the imported value.\n   */\n  bindImport(\n    moduleSpecifier: string,\n    exportedName: string,\n    target: WalkerPath<ASTv1.Node>,\n    opts?: { nameHint?: string }\n  ): string {\n    // This will discover or create the local name for accessing the given import.\n    let importedIdentifier = this.#importer.import(\n      this.#template,\n      moduleSpecifier,\n      exportedName,\n      opts?.nameHint\n    );\n\n    let identifier = unusedNameLike(importedIdentifier.name, (candidate) =>\n      astNodeHasBinding(target, candidate)\n    );\n    if (identifier !== importedIdentifier.name) {\n      // The importedIdentifier that we have in Javascript is not usable within\n      // our HBS because it's shadowed by a block param. So we will introduce a\n      // second name via a variable declaration.\n      //\n      // The reason we don't force the import itself to have this name is that\n      // we might be re-using an existing import, and we don't want to go\n      // rewriting all of its callsites that are unrelated to us.\n      let t = this.#babel.types;\n      this.#emitStatement(\n        t.variableDeclaration('let', [\n          t.variableDeclarator(t.identifier(identifier), importedIdentifier),\n        ])\n      );\n    }\n    this.#locals.push(identifier);\n    return identifier;\n  }\n\n  /**\n   * Add an import statement purely for side effect.\n   *\n   * @param moduleSpecifier the module to import\n   */\n  importForSideEffect(moduleSpecifier: string): void {\n    this.#importer.importForSideEffect(moduleSpecifier);\n  }\n\n  /**\n   * Emit a javascript expresison for side-effect. This only accepts\n   * expressions, not statements, because you should not introduce new bindings.\n   * To introduce a binding see bindExpression or bindImport instead.\n   *\n   * @param { Expression } expression A javascript expression whose value will\n   * initialize your new binding. See docs on the Expression type below for\n   * details.\n   */\n  emitExpression(expression: Expression): void {\n    let t = this.#babel.types;\n    this.#emitStatement(t.expressionStatement(this.#parseExpression(this.#program, expression)));\n  }\n\n  #parseExpression(target: NodePath<t.Node>, expression: Expression): t.Expression {\n    let expressionString: string;\n    if (typeof expression === 'string') {\n      expressionString = expression;\n    } else {\n      expressionString = expression(new ExpressionContext(this.#importer, target));\n    }\n\n    let parsed = this.#babel.parse(expressionString);\n    if (!parsed) {\n      throw new Error(\n        `JSUtils.bindExpression could not understand the expression: ${expressionString}`\n      );\n    }\n    let statements = body(parsed);\n    if (statements.length !== 1) {\n      throw new Error(\n        `JSUtils.bindExpression expected to find exactly one expression but found ${statements.length} in: ${expressionString}`\n      );\n    }\n    let statement = statements[0];\n    if (statement.type !== 'ExpressionStatement') {\n      throw new Error(\n        `JSUtils.bindExpression expected to find an expression but found ${statement.type} in: ${expressionString}`\n      );\n    }\n    return statement.expression;\n  }\n}\n\nfunction unusedNameLike(desiredName: string, isUsed: (name: string) => boolean): string {\n  let candidate = desiredName;\n  let counter = 0;\n  while (isUsed(candidate)) {\n    candidate = `${desiredName}${counter++}`;\n  }\n  return candidate;\n}\n\nfunction astNodeHasBinding(target: WalkerPath<ASTv1.Node>, name: string): boolean {\n  let cursor: WalkerPath<ASTv1.Node> | null = target;\n  while (cursor) {\n    let parentNode = cursor.parent?.node;\n    if (\n      parentNode?.type === 'ElementNode' &&\n      parentNode.blockParams.includes(name) &&\n      // an ElementNode's block params are valid only within its children\n      parentNode.children.includes(cursor.node as ASTv1.Statement)\n    ) {\n      return true;\n    }\n\n    if (\n      parentNode?.type === 'Block' &&\n      parentNode.blockParams.includes(name) &&\n      // a Block's blockParams are valid only within its body\n      parentNode.body.includes(cursor.node as ASTv1.Statement)\n    ) {\n      return true;\n    }\n\n    cursor = cursor.parent;\n  }\n  return false;\n}\n\n/**\n * This extends Glimmer's ASTPluginEnvironment type to put our jsutils into meta\n */\nexport type WithJSUtils<T extends { meta?: object }> = {\n  meta: T['meta'] & { jsutils: JSUtils };\n} & T;\n\nfunction body(node: t.Program | t.File) {\n  if (node.type === 'File') {\n    return node.program.body;\n  } else {\n    return node.body;\n  }\n}\n\n/**\n * Allows you to construct an expression that relies on imported values.\n */\nclass ExpressionContext {\n  #importer: ImportUtil;\n  #target: NodePath<t.Node>;\n\n  constructor(importer: ImportUtil, target: NodePath<t.Node>) {\n    this.#importer = importer;\n    this.#target = target;\n  }\n\n  /**\n   * Find or create a local binding for the given import.\n   *\n   * @param moduleSpecifier The path to import from.\n   * @param exportedName The named export you wish to access, or \"default\" for\n   * the default export, or \"*\" for the namespace export.\n   * @param nameHint Optionally, provide a descriptive name for your new\n   * binding. We will mangle this name as needed to avoid collisions, but\n   * picking a good name here can aid in debugging.\n\n   * @return the local identifier for the imported value\n   */\n  import(moduleSpecifier: string, exportedName: string, nameHint?: string): string {\n    return this.#importer.import(this.#target, moduleSpecifier, exportedName, nameHint).name;\n  }\n}\n\n/**\n * You can pass a Javascript expression as a string like:\n *\n *   \"new Date()\"\n *\n * Or as a function that returns a string:\n *\n *   () => \"new Date()\"\n *\n * When you use a function, it can use imported values:\n *\n *   (context) => `new ${context.import(\"luxon\", \"DateTime\")}()`\n *\n */\nexport type Expression = string | ((context: ExpressionContext) => string);\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"js-utils.js","sourceRoot":"","sources":["js-utils.ts"],"names":[],"mappings":"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","sourcesContent":["import type { types as t } from '@babel/core';\nimport type * as Babel from '@babel/core';\nimport type { NodePath } from '@babel/traverse';\nimport type { ASTPluginBuilder, ASTPluginEnvironment, ASTv1, WalkerPath } from '@glimmer/syntax';\nimport type { ImportUtil } from 'babel-import-util';\nimport type { State } from './plugin';\n\n// This exists to give AST plugins a controlled interface for influencing the\n// surrounding Javascript scope\nexport class JSUtils {\n  #babel: typeof Babel;\n  #state: State<unknown>;\n  #template: NodePath<t.Expression>;\n  #locals: string[];\n  #importer: ImportUtil;\n\n  constructor(\n    babel: typeof Babel,\n    state: State<unknown>,\n    template: NodePath<t.Expression>,\n    locals: string[],\n    importer: ImportUtil\n  ) {\n    this.#babel = babel;\n    this.#state = state;\n    this.#template = template;\n    this.#locals = locals;\n    this.#importer = importer;\n  }\n\n  /**\n   * Create a new binding that you can use in your template, initialized with\n   * the given Javascript expression.\n   *\n   * @param { Expression } expression A javascript expression whose value will\n   * initialize your new binding. See docs on the Expression type for details.\n   * @param target The location within your template where the binding will be\n   * used. This matters so we can avoid naming collisions.\n   * @param opts.nameHint Optionally, provide a descriptive name for your new\n   * binding. We will mangle this name as needed to avoid collisions, but\n   * picking a good name here can aid in debugging.\n   *\n   * @return The name you can use in your template to access the binding.\n   */\n  bindExpression(\n    expression: Expression,\n    target: WalkerPath<ASTv1.Node>,\n    opts?: { nameHint?: string }\n  ): string {\n    let name = unusedNameLike(\n      opts?.nameHint ?? 'a',\n      (candidate) =>\n        this.#template.scope.hasBinding(candidate) || astNodeHasBinding(target, candidate)\n    );\n    let t = this.#babel.types;\n    let declaration: NodePath<t.VariableDeclaration> = this.#emitStatement(\n      t.variableDeclaration('let', [\n        t.variableDeclarator(\n          t.identifier(name),\n          this.#parseExpression(this.#state.program, expression)\n        ),\n      ])\n    );\n    declaration.scope.registerBinding('module', declaration.get('declarations.0') as NodePath);\n    this.#locals.push(name);\n    return name;\n  }\n\n  #emitStatement<T extends t.Statement>(statement: T): NodePath<T> {\n    if (this.#state.lastInsertedPath) {\n      this.#state.lastInsertedPath = this.#state.lastInsertedPath.insertAfter(statement)[0];\n    } else {\n      this.#state.lastInsertedPath = this.#state.program.unshiftContainer('body', statement)[0];\n    }\n    return this.#state.lastInsertedPath as NodePath<T>;\n  }\n\n  /**\n   * Gain access to an imported value within your template.\n   *\n   * @param moduleSpecifier The path to import from.\n   * @param exportedName The named export you wish to access, or \"default\" for\n   * the default export, or \"*\" for the namespace export.\n   * @param target The location within your template where the binding will be\n   * used. This matters so we can avoid naming collisions.\n   * @param opts.nameHint Optionally, provide a descriptive name for your new\n   * binding. We will mangle this name as needed to avoid collisions, but\n   * picking a good name here can aid in debugging.\n   *\n   * @return The name you can use in your template to access the imported value.\n   */\n  bindImport(\n    moduleSpecifier: string,\n    exportedName: string,\n    target: WalkerPath<ASTv1.Node>,\n    opts?: { nameHint?: string }\n  ): string {\n    // This will discover or create the local name for accessing the given import.\n    let importedIdentifier = this.#importer.import(\n      this.#template,\n      moduleSpecifier,\n      exportedName,\n      opts?.nameHint\n    );\n\n    let identifier = unusedNameLike(importedIdentifier.name, (candidate) =>\n      astNodeHasBinding(target, candidate)\n    );\n    if (identifier !== importedIdentifier.name) {\n      // The importedIdentifier that we have in Javascript is not usable within\n      // our HBS because it's shadowed by a block param. So we will introduce a\n      // second name via a variable declaration.\n      //\n      // The reason we don't force the import itself to have this name is that\n      // we might be re-using an existing import, and we don't want to go\n      // rewriting all of its callsites that are unrelated to us.\n      let t = this.#babel.types;\n      this.#emitStatement(\n        t.variableDeclaration('let', [\n          t.variableDeclarator(t.identifier(identifier), importedIdentifier),\n        ])\n      );\n    }\n    this.#locals.push(identifier);\n    return identifier;\n  }\n\n  /**\n   * Add an import statement purely for side effect.\n   *\n   * @param moduleSpecifier the module to import\n   */\n  importForSideEffect(moduleSpecifier: string): void {\n    this.#importer.importForSideEffect(moduleSpecifier);\n  }\n\n  /**\n   * Emit a javascript expresison for side-effect. This only accepts\n   * expressions, not statements, because you should not introduce new bindings.\n   * To introduce a binding see bindExpression or bindImport instead.\n   *\n   * @param { Expression } expression A javascript expression whose value will\n   * initialize your new binding. See docs on the Expression type below for\n   * details.\n   */\n  emitExpression(expression: Expression): void {\n    let t = this.#babel.types;\n    this.#emitStatement(\n      t.expressionStatement(this.#parseExpression(this.#state.program, expression))\n    );\n  }\n\n  #parseExpression(target: NodePath<t.Node>, expression: Expression): t.Expression {\n    let expressionString: string;\n    if (typeof expression === 'string') {\n      expressionString = expression;\n    } else {\n      expressionString = expression(new ExpressionContext(this.#importer, target));\n    }\n\n    let parsed = this.#babel.parse(expressionString);\n    if (!parsed) {\n      throw new Error(\n        `JSUtils.bindExpression could not understand the expression: ${expressionString}`\n      );\n    }\n    let statements = body(parsed);\n    if (statements.length !== 1) {\n      throw new Error(\n        `JSUtils.bindExpression expected to find exactly one expression but found ${statements.length} in: ${expressionString}`\n      );\n    }\n    let statement = statements[0];\n    if (statement.type !== 'ExpressionStatement') {\n      throw new Error(\n        `JSUtils.bindExpression expected to find an expression but found ${statement.type} in: ${expressionString}`\n      );\n    }\n    return statement.expression;\n  }\n}\n\nfunction unusedNameLike(desiredName: string, isUsed: (name: string) => boolean): string {\n  let candidate = desiredName;\n  let counter = 0;\n  while (isUsed(candidate)) {\n    candidate = `${desiredName}${counter++}`;\n  }\n  return candidate;\n}\n\nfunction astNodeHasBinding(target: WalkerPath<ASTv1.Node>, name: string): boolean {\n  let cursor: WalkerPath<ASTv1.Node> | null = target;\n  while (cursor) {\n    let parentNode = cursor.parent?.node;\n    if (\n      parentNode?.type === 'ElementNode' &&\n      parentNode.blockParams.includes(name) &&\n      // an ElementNode's block params are valid only within its children\n      parentNode.children.includes(cursor.node as ASTv1.Statement)\n    ) {\n      return true;\n    }\n\n    if (\n      parentNode?.type === 'Block' &&\n      parentNode.blockParams.includes(name) &&\n      // a Block's blockParams are valid only within its body\n      parentNode.body.includes(cursor.node as ASTv1.Statement)\n    ) {\n      return true;\n    }\n\n    cursor = cursor.parent;\n  }\n  return false;\n}\n\n/**\n * This extends Glimmer's ASTPluginEnvironment type to put our jsutils into meta\n */\nexport type WithJSUtils<T extends { meta?: object }> = {\n  meta: T['meta'] & { jsutils: JSUtils };\n} & T;\n\nexport type ExtendedPluginBuilder = ASTPluginBuilder<WithJSUtils<ASTPluginEnvironment>>;\n\nfunction body(node: t.Program | t.File) {\n  if (node.type === 'File') {\n    return node.program.body;\n  } else {\n    return node.body;\n  }\n}\n\n/**\n * Allows you to construct an expression that relies on imported values.\n */\nclass ExpressionContext {\n  #importer: ImportUtil;\n  #target: NodePath<t.Node>;\n\n  constructor(importer: ImportUtil, target: NodePath<t.Node>) {\n    this.#importer = importer;\n    this.#target = target;\n  }\n\n  /**\n   * Find or create a local binding for the given import.\n   *\n   * @param moduleSpecifier The path to import from.\n   * @param exportedName The named export you wish to access, or \"default\" for\n   * the default export, or \"*\" for the namespace export.\n   * @param nameHint Optionally, provide a descriptive name for your new\n   * binding. We will mangle this name as needed to avoid collisions, but\n   * picking a good name here can aid in debugging.\n\n   * @return the local identifier for the imported value\n   */\n  import(moduleSpecifier: string, exportedName: string, nameHint?: string): string {\n    return this.#importer.import(this.#target, moduleSpecifier, exportedName, nameHint).name;\n  }\n}\n\n/**\n * You can pass a Javascript expression as a string like:\n *\n *   \"new Date()\"\n *\n * Or as a function that returns a string:\n *\n *   () => \"new Date()\"\n *\n * When you use a function, it can use imported values:\n *\n *   (context) => `new ${context.import(\"luxon\", \"DateTime\")}()`\n *\n */\nexport type Expression = string | ((context: ExpressionContext) => string);\n"]} |
@@ -1,14 +0,15 @@ | ||
import type * as Babel from '@babel/core'; | ||
import { Options as PluginOptions, EmberPrecompile } from './plugin'; | ||
export interface Options extends PluginOptions { | ||
precompilerPath?: string; | ||
precompile?: EmberPrecompile; | ||
} | ||
declare const htmlbarsInlinePrecompile: { | ||
(babel: typeof Babel): Babel.PluginObj<Options>; | ||
import { Options as SharedOptions } from './plugin'; | ||
import { EmberTemplateCompiler } from './ember-template-compiler'; | ||
import { ExtendedPluginBuilder } from './js-utils'; | ||
export declare type Options = Omit<SharedOptions, 'transforms' | 'compiler'> & { | ||
compilerPath?: string; | ||
compiler?: EmberTemplateCompiler; | ||
transforms?: (ExtendedPluginBuilder | string)[]; | ||
}; | ||
declare const _default: ((babel: typeof import("@babel/core")) => import("@babel/core").PluginObj<import("./plugin").State<Options>>) & { | ||
baseDir(): string; | ||
_parallelBabel: { | ||
requireFile: string; | ||
}; | ||
baseDir(): string; | ||
}; | ||
export default htmlbarsInlinePrecompile; | ||
export default _default; |
@@ -8,12 +8,32 @@ "use strict"; | ||
const plugin_1 = __importDefault(require("./plugin")); | ||
const htmlbarsInlinePrecompile = plugin_1.default(function (opts) { | ||
if (opts.precompilerPath) { | ||
const ember_template_compiler_1 = require("./ember-template-compiler"); | ||
function handleNodeSpecificOptions(opts) { | ||
let compiler; | ||
if (opts.compilerPath) { | ||
// eslint-disable-next-line @typescript-eslint/no-var-requires | ||
let mod = require(opts.precompilerPath); | ||
return mod.precompile; | ||
let mod = require(opts.compilerPath); | ||
ember_template_compiler_1.assertTemplateCompiler(mod); | ||
compiler = mod; | ||
} | ||
else if (opts.precompile) { | ||
return opts.precompile; | ||
else if (opts.compiler) { | ||
ember_template_compiler_1.assertTemplateCompiler(opts.compiler); | ||
compiler = opts.compiler; | ||
} | ||
}); | ||
else { | ||
throw new Error(`must provide compilerPath or compiler`); | ||
} | ||
let transforms = []; | ||
if (opts.transforms) { | ||
transforms = opts.transforms.map((t) => { | ||
if (typeof t === 'string') { | ||
return require(t); | ||
} | ||
else { | ||
return t; | ||
} | ||
}); | ||
} | ||
return Object.assign(Object.assign({}, opts), { transforms, compiler }); | ||
} | ||
const htmlbarsInlinePrecompile = plugin_1.default(handleNodeSpecificOptions); | ||
htmlbarsInlinePrecompile._parallelBabel = { | ||
@@ -26,2 +46,2 @@ requireFile: __filename, | ||
exports.default = htmlbarsInlinePrecompile; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibm9kZS1tYWluLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsibm9kZS1tYWluLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsK0JBQStCO0FBQy9CLHNEQUFrQztBQXNCbEMsTUFBTSx3QkFBd0IsR0FBRyxnQkFBVSxDQUFDLFVBQVUsSUFBYTtJQUNqRSxJQUFJLElBQUksQ0FBQyxlQUFlLEVBQUU7UUFDeEIsOERBQThEO1FBQzlELElBQUksR0FBRyxHQUFRLE9BQU8sQ0FBQyxJQUFJLENBQUMsZUFBZSxDQUFDLENBQUM7UUFDN0MsT0FBTyxHQUFHLENBQUMsVUFBVSxDQUFDO0tBQ3ZCO1NBQU0sSUFBSSxJQUFJLENBQUMsVUFBVSxFQUFFO1FBQzFCLE9BQU8sSUFBSSxDQUFDLFVBQVUsQ0FBQztLQUN4QjtBQUNILENBQUMsQ0FJQSxDQUFDO0FBRUYsd0JBQXdCLENBQUMsY0FBYyxHQUFHO0lBQ3hDLFdBQVcsRUFBRSxVQUFVO0NBQ3hCLENBQUM7QUFFRix3QkFBd0IsQ0FBQyxPQUFPLEdBQUc7SUFDakMsT0FBTyxjQUFPLENBQUMsU0FBUyxFQUFFLElBQUksQ0FBQyxDQUFDO0FBQ2xDLENBQUMsQ0FBQztBQUVGLGtCQUFlLHdCQUF3QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgcmVzb2x2ZSB9IGZyb20gJ3BhdGgnO1xuaW1wb3J0IG1ha2VQbHVnaW4gZnJvbSAnLi9wbHVnaW4nO1xuaW1wb3J0IHR5cGUgKiBhcyBCYWJlbCBmcm9tICdAYmFiZWwvY29yZSc7XG5cbmltcG9ydCB7IE9wdGlvbnMgYXMgUGx1Z2luT3B0aW9ucywgRW1iZXJQcmVjb21waWxlIH0gZnJvbSAnLi9wbHVnaW4nO1xuXG5leHBvcnQgaW50ZXJmYWNlIE9wdGlvbnMgZXh0ZW5kcyBQbHVnaW5PcHRpb25zIHtcbiAgLy8gVGhlIG9uLWRpc2sgcGF0aCB0byBhIG1vZHVsZSB0aGF0IHByb3ZpZGVzIGEgYHByZWNvbXBpbGVgIGZ1bmN0aW9uIGFzXG4gIC8vIGRlZmluZWQgYmVsb3cuIFlvdSBuZWVkIHRvIGVpdGhlciBzZXQgYHByZWNvbXBpbGVQYXRoYCBvciBzZXQgYHByZWNvbXBpbGVgLlxuICBwcmVjb21waWxlclBhdGg/OiBzdHJpbmc7XG5cbiAgLy8gQSBwcmVjb21waWxlIGZ1bmN0aW9uIHRoYXQgaW52b2tlcyBFbWJlcidzIHRlbXBsYXRlIGNvbXBpbGVyLlxuICAvL1xuICAvLyBPcHRpb25zIGhhbmRsaW5nIHJ1bGVzOlxuICAvL1xuICAvLyAgLSB3ZSBhZGQgYGNvbnRlbnRgLCB3aGljaCBpcyB0aGUgb3JpZ2luYWwgc3RyaW5nIGZvcm0gb2YgdGhlIHRlbXBsYXRlXG4gIC8vICAtIHdlIGhhdmUgc3BlY2lhbCBwYXJzaW5nIGZvciBgc2NvcGVgIHdoaWNoIGJlY29tZXMgYGxvY2Fsc2Agd2hlbiBwYXNzZWRcbiAgLy8gICAgdG8geW91ciBwcmVjb21waWxlXG4gIC8vICAtIGFueXRoaW5nIGVsc2UgdGhlIHVzZXIgcGFzc2VzIHRvIGBwcmVjb21waWxlVGVtcGxhdGVgIHdpbGwgYmUgcGFzc2VkXG4gIC8vICAgIHRocm91Z2ggdG8geW91ciBgcHJlY29tcGlsZWAuXG4gIHByZWNvbXBpbGU/OiBFbWJlclByZWNvbXBpbGU7XG59XG5cbmNvbnN0IGh0bWxiYXJzSW5saW5lUHJlY29tcGlsZSA9IG1ha2VQbHVnaW4oZnVuY3Rpb24gKG9wdHM6IE9wdGlvbnMpIHtcbiAgaWYgKG9wdHMucHJlY29tcGlsZXJQYXRoKSB7XG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby12YXItcmVxdWlyZXNcbiAgICBsZXQgbW9kOiBhbnkgPSByZXF1aXJlKG9wdHMucHJlY29tcGlsZXJQYXRoKTtcbiAgICByZXR1cm4gbW9kLnByZWNvbXBpbGU7XG4gIH0gZWxzZSBpZiAob3B0cy5wcmVjb21waWxlKSB7XG4gICAgcmV0dXJuIG9wdHMucHJlY29tcGlsZTtcbiAgfVxufSkgYXMge1xuICAoYmFiZWw6IHR5cGVvZiBCYWJlbCk6IEJhYmVsLlBsdWdpbk9iajxPcHRpb25zPjtcbiAgX3BhcmFsbGVsQmFiZWw6IHsgcmVxdWlyZUZpbGU6IHN0cmluZyB9O1xuICBiYXNlRGlyKCk6IHN0cmluZztcbn07XG5cbmh0bWxiYXJzSW5saW5lUHJlY29tcGlsZS5fcGFyYWxsZWxCYWJlbCA9IHtcbiAgcmVxdWlyZUZpbGU6IF9fZmlsZW5hbWUsXG59O1xuXG5odG1sYmFyc0lubGluZVByZWNvbXBpbGUuYmFzZURpciA9IGZ1bmN0aW9uICgpIHtcbiAgcmV0dXJuIHJlc29sdmUoX19kaXJuYW1lLCAnLi4nKTtcbn07XG5cbmV4cG9ydCBkZWZhdWx0IGh0bWxiYXJzSW5saW5lUHJlY29tcGlsZTtcbiJdfQ== | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibm9kZS1tYWluLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsibm9kZS1tYWluLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsK0JBQStCO0FBQy9CLHNEQUFrQztBQUdsQyx1RUFBMEY7QUFpQjFGLFNBQVMseUJBQXlCLENBQUMsSUFBYTtJQUM5QyxJQUFJLFFBQStCLENBQUM7SUFDcEMsSUFBSSxJQUFJLENBQUMsWUFBWSxFQUFFO1FBQ3JCLDhEQUE4RDtRQUM5RCxJQUFJLEdBQUcsR0FBUSxPQUFPLENBQUMsSUFBSSxDQUFDLFlBQVksQ0FBQyxDQUFDO1FBQzFDLGdEQUFzQixDQUFDLEdBQUcsQ0FBQyxDQUFDO1FBQzVCLFFBQVEsR0FBRyxHQUFHLENBQUM7S0FDaEI7U0FBTSxJQUFJLElBQUksQ0FBQyxRQUFRLEVBQUU7UUFDeEIsZ0RBQXNCLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBQ3RDLFFBQVEsR0FBRyxJQUFJLENBQUMsUUFBUSxDQUFDO0tBQzFCO1NBQU07UUFDTCxNQUFNLElBQUksS0FBSyxDQUFDLHVDQUF1QyxDQUFDLENBQUM7S0FDMUQ7SUFFRCxJQUFJLFVBQVUsR0FBRyxFQUFFLENBQUM7SUFDcEIsSUFBSSxJQUFJLENBQUMsVUFBVSxFQUFFO1FBQ25CLFVBQVUsR0FBRyxJQUFJLENBQUMsVUFBVSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFO1lBQ3JDLElBQUksT0FBTyxDQUFDLEtBQUssUUFBUSxFQUFFO2dCQUN6QixPQUFPLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQzthQUNuQjtpQkFBTTtnQkFDTCxPQUFPLENBQUMsQ0FBQzthQUNWO1FBQ0gsQ0FBQyxDQUFDLENBQUM7S0FDSjtJQUNELHVDQUFZLElBQUksS0FBRSxVQUFVLEVBQUUsUUFBUSxJQUFHO0FBQzNDLENBQUM7QUFFRCxNQUFNLHdCQUF3QixHQUFHLGdCQUFVLENBQUMseUJBQXlCLENBQUMsQ0FBQztBQUV0RSx3QkFBZ0MsQ0FBQyxjQUFjLEdBQUc7SUFDakQsV0FBVyxFQUFFLFVBQVU7Q0FDeEIsQ0FBQztBQUVELHdCQUFnQyxDQUFDLE9BQU8sR0FBRztJQUMxQyxPQUFPLGNBQU8sQ0FBQyxTQUFTLEVBQUUsSUFBSSxDQUFDLENBQUM7QUFDbEMsQ0FBQyxDQUFDO0FBRUYsa0JBQWUsd0JBR2QsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHJlc29sdmUgfSBmcm9tICdwYXRoJztcbmltcG9ydCBtYWtlUGx1Z2luIGZyb20gJy4vcGx1Z2luJztcblxuaW1wb3J0IHsgT3B0aW9ucyBhcyBTaGFyZWRPcHRpb25zIH0gZnJvbSAnLi9wbHVnaW4nO1xuaW1wb3J0IHsgYXNzZXJ0VGVtcGxhdGVDb21waWxlciwgRW1iZXJUZW1wbGF0ZUNvbXBpbGVyIH0gZnJvbSAnLi9lbWJlci10ZW1wbGF0ZS1jb21waWxlcic7XG5pbXBvcnQgeyBFeHRlbmRlZFBsdWdpbkJ1aWxkZXIgfSBmcm9tICcuL2pzLXV0aWxzJztcblxuZXhwb3J0IHR5cGUgT3B0aW9ucyA9IE9taXQ8U2hhcmVkT3B0aW9ucywgJ3RyYW5zZm9ybXMnIHwgJ2NvbXBpbGVyJz4gJiB7XG4gIC8vIFRoZSBvbi1kaXNrIHBhdGggdG8gdGhlIGVtYmVyLXRlbXBsYXRlLWNvbWlwbGVyLmpzIG1vZHVsZSBmb3Igb3VyIGN1cnJlbnRcbiAgLy8gZW1iZXIgdmVyc2lvbi4gWW91IG5lZWQgdG8gZWl0aGVyIHNldCBgY29tcGlsZXJQYXRoYCBvciBzZXQgYGNvbXBpbGVyYC5cbiAgY29tcGlsZXJQYXRoPzogc3RyaW5nO1xuXG4gIC8vIFRoZSBlbWJlci10ZW1wbGF0ZS1jb21waWxlci5qcyBtb2R1bGUgdGhhdCBzaGlwcyB3aXRoaW4geW91ciBlbWJlci1zb3VyY2VcbiAgLy8gdmVyc2lvbi4gWW91IG5lZWQgdG8gc2V0IGVpdGhlciBgY29tcGlsZXJQYXRoYCBvciBgY29tcGlsZXJgLlxuICBjb21waWxlcj86IEVtYmVyVGVtcGxhdGVDb21waWxlcjtcblxuICAvLyBMaXN0IG9mIGN1c3RvbSB0cmFuc2Zvcm1hdGlvbnMgdG8gYXBwbHkgdG8gdGhlIGhhbmRsZWJhcnMgQVNUIGJlZm9yZVxuICAvLyBjb21waWxhdGlvbi4gVGhlc2UgY2FuIGJlIHRoZSBhY3R1YWwgZnVuY3Rpb25zIG9yIHJlc29sdmFibGUgbW9kdWxlIG5hbWVzLlxuICB0cmFuc2Zvcm1zPzogKEV4dGVuZGVkUGx1Z2luQnVpbGRlciB8IHN0cmluZylbXTtcbn07XG5cbmZ1bmN0aW9uIGhhbmRsZU5vZGVTcGVjaWZpY09wdGlvbnMob3B0czogT3B0aW9ucyk6IFNoYXJlZE9wdGlvbnMge1xuICBsZXQgY29tcGlsZXI6IEVtYmVyVGVtcGxhdGVDb21waWxlcjtcbiAgaWYgKG9wdHMuY29tcGlsZXJQYXRoKSB7XG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby12YXItcmVxdWlyZXNcbiAgICBsZXQgbW9kOiBhbnkgPSByZXF1aXJlKG9wdHMuY29tcGlsZXJQYXRoKTtcbiAgICBhc3NlcnRUZW1wbGF0ZUNvbXBpbGVyKG1vZCk7XG4gICAgY29tcGlsZXIgPSBtb2Q7XG4gIH0gZWxzZSBpZiAob3B0cy5jb21waWxlcikge1xuICAgIGFzc2VydFRlbXBsYXRlQ29tcGlsZXIob3B0cy5jb21waWxlcik7XG4gICAgY29tcGlsZXIgPSBvcHRzLmNvbXBpbGVyO1xuICB9IGVsc2Uge1xuICAgIHRocm93IG5ldyBFcnJvcihgbXVzdCBwcm92aWRlIGNvbXBpbGVyUGF0aCBvciBjb21waWxlcmApO1xuICB9XG5cbiAgbGV0IHRyYW5zZm9ybXMgPSBbXTtcbiAgaWYgKG9wdHMudHJhbnNmb3Jtcykge1xuICAgIHRyYW5zZm9ybXMgPSBvcHRzLnRyYW5zZm9ybXMubWFwKCh0KSA9PiB7XG4gICAgICBpZiAodHlwZW9mIHQgPT09ICdzdHJpbmcnKSB7XG4gICAgICAgIHJldHVybiByZXF1aXJlKHQpO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgcmV0dXJuIHQ7XG4gICAgICB9XG4gICAgfSk7XG4gIH1cbiAgcmV0dXJuIHsgLi4ub3B0cywgdHJhbnNmb3JtcywgY29tcGlsZXIgfTtcbn1cblxuY29uc3QgaHRtbGJhcnNJbmxpbmVQcmVjb21waWxlID0gbWFrZVBsdWdpbihoYW5kbGVOb2RlU3BlY2lmaWNPcHRpb25zKTtcblxuKGh0bWxiYXJzSW5saW5lUHJlY29tcGlsZSBhcyBhbnkpLl9wYXJhbGxlbEJhYmVsID0ge1xuICByZXF1aXJlRmlsZTogX19maWxlbmFtZSxcbn07XG5cbihodG1sYmFyc0lubGluZVByZWNvbXBpbGUgYXMgYW55KS5iYXNlRGlyID0gZnVuY3Rpb24gKCkge1xuICByZXR1cm4gcmVzb2x2ZShfX2Rpcm5hbWUsICcuLicpO1xufTtcblxuZXhwb3J0IGRlZmF1bHQgaHRtbGJhcnNJbmxpbmVQcmVjb21waWxlIGFzIHR5cGVvZiBodG1sYmFyc0lubGluZVByZWNvbXBpbGUgJiB7XG4gIGJhc2VEaXIoKTogc3RyaW5nO1xuICBfcGFyYWxsZWxCYWJlbDogeyByZXF1aXJlRmlsZTogc3RyaW5nIH07XG59O1xuIl19 |
@@ -5,12 +5,16 @@ import type { NodePath } from '@babel/traverse'; | ||
import { ImportUtil } from 'babel-import-util'; | ||
export declare type EmberPrecompile = (templateString: string, options: Record<string, unknown>) => string; | ||
import { ExtendedPluginBuilder } from './js-utils'; | ||
import type { EmberTemplateCompiler } from './ember-template-compiler'; | ||
export declare type LegacyModuleName = 'ember-cli-htmlbars' | 'ember-cli-htmlbars-inline-precompile' | 'htmlbars-inline-precompile'; | ||
export interface Options { | ||
compiler: EmberTemplateCompiler; | ||
outputModuleOverrides?: Record<string, Record<string, [string, string]>>; | ||
enableLegacyModules?: LegacyModuleName[]; | ||
targetFormat?: 'wire' | 'hbs'; | ||
transforms?: ExtendedPluginBuilder[]; | ||
} | ||
interface State { | ||
opts: Options; | ||
export interface State<EnvSpecificOptions> { | ||
opts: EnvSpecificOptions; | ||
normalizedOpts: Required<Options>; | ||
util: ImportUtil; | ||
precompile: EmberPrecompile; | ||
templateFactory: { | ||
@@ -21,4 +25,5 @@ moduleName: string; | ||
program: NodePath<t.Program>; | ||
lastInsertedPath: NodePath<t.Statement> | undefined; | ||
filename: string; | ||
} | ||
export default function makePlugin<O>(loadPrecompiler: (opts: O) => EmberPrecompile): (babel: typeof Babel) => Babel.PluginObj<State>; | ||
export {}; | ||
export default function makePlugin<EnvSpecificOptions>(loadOptions: (opts: EnvSpecificOptions) => Options): (babel: typeof Babel) => Babel.PluginObj<State<EnvSpecificOptions>>; |
@@ -32,39 +32,5 @@ "use strict"; | ||
]; | ||
function makePlugin( | ||
// receives the Babel plugin options, returns Ember's precompiler | ||
loadPrecompiler) { | ||
function makePlugin(loadOptions) { | ||
return function htmlbarsInlinePrecompile(babel) { | ||
let t = babel.types; | ||
function insertCompiledTemplate(target, state, template, userTypedOptions) { | ||
if (!userTypedOptions.locals) { | ||
userTypedOptions.locals = []; | ||
} | ||
let jsutils = new js_utils_1.JSUtils(babel, state.program, target, userTypedOptions.locals, state.util); | ||
let meta = Object.assign({ jsutils }, userTypedOptions === null || userTypedOptions === void 0 ? void 0 : userTypedOptions.meta); | ||
let options = Object.assign({}, userTypedOptions, { contents: template, meta }); | ||
let precompile = state.precompile; | ||
let precompileResultString; | ||
if (options.insertRuntimeErrors) { | ||
try { | ||
precompileResultString = precompile(template, options); | ||
} | ||
catch (error) { | ||
target.replaceWith(runtimeErrorIIFE(babel, { ERROR_MESSAGE: error.message })); | ||
return; | ||
} | ||
} | ||
else { | ||
precompileResultString = precompile(template, options); | ||
} | ||
let precompileResultAST = babel.parse(`var precompileResult = ${precompileResultString};`, { | ||
babelrc: false, | ||
configFile: false, | ||
}); | ||
let templateExpression = precompileResultAST.program.body[0] | ||
.declarations[0].init; | ||
t.addComment(templateExpression, 'leading', `\n ${template.replace(/\*\//g, '*\\/')}\n`, | ||
/* line comment? */ false); | ||
let templateFactoryIdentifier = state.util.import(target, state.templateFactory.moduleName, state.templateFactory.exportName); | ||
target.replaceWith(t.callExpression(templateFactoryIdentifier, [templateExpression])); | ||
} | ||
return { | ||
@@ -74,16 +40,12 @@ visitor: { | ||
enter(path, state) { | ||
var _a, _b; | ||
let moduleName = '@ember/template-factory'; | ||
let exportName = 'createTemplateFactory'; | ||
let overrides = (_b = (_a = state.opts.outputModuleOverrides) === null || _a === void 0 ? void 0 : _a[moduleName]) === null || _b === void 0 ? void 0 : _b[exportName]; | ||
state.templateFactory = overrides | ||
? { exportName: overrides[0], moduleName: overrides[1] } | ||
: { exportName, moduleName }; | ||
state.normalizedOpts = Object.assign({ targetFormat: 'wire', outputModuleOverrides: {}, enableLegacyModules: [], transforms: [] }, loadOptions(state.opts)); | ||
state.templateFactory = templateFactoryConfig(state.normalizedOpts); | ||
state.util = new babel_import_util_1.ImportUtil(t, path); | ||
state.precompile = loadPrecompiler(state.opts); | ||
state.program = path; | ||
}, | ||
exit(_path, state) { | ||
for (let { moduleName, export: exportName } of configuredModules(state)) { | ||
state.util.removeImport(moduleName, exportName); | ||
if (state.normalizedOpts.targetFormat === 'wire') { | ||
for (let { moduleName, export: exportName } of configuredModules(state)) { | ||
state.util.removeImport(moduleName, exportName); | ||
} | ||
} | ||
@@ -108,3 +70,8 @@ }, | ||
let template = path.node.quasi.quasis.map((quasi) => quasi.value.cooked).join(''); | ||
insertCompiledTemplate(path, state, template, {}); | ||
if (state.normalizedOpts.targetFormat === 'wire') { | ||
insertCompiledTemplate(babel, state, template, path, {}); | ||
} | ||
else { | ||
insertTransformedTemplate(babel, state, template, path, {}, options); | ||
} | ||
}, | ||
@@ -147,3 +114,3 @@ CallExpression(path, state) { | ||
} | ||
userTypedOptions = new expression_parser_1.ExpressionParser(babel).parseObjectExpression(calleePath.node.name, secondArg, true); | ||
userTypedOptions = new expression_parser_1.ExpressionParser(babel).parseObjectExpression(calleePath.node.name, secondArg, options.enableScope); | ||
} | ||
@@ -153,3 +120,8 @@ if (restArgs.length > 0) { | ||
} | ||
insertCompiledTemplate(path, state, template, userTypedOptions); | ||
if (state.normalizedOpts.targetFormat === 'wire') { | ||
insertCompiledTemplate(babel, state, template, path, userTypedOptions); | ||
} | ||
else { | ||
insertTransformedTemplate(babel, state, template, path, userTypedOptions, options); | ||
} | ||
}, | ||
@@ -162,6 +134,5 @@ }, | ||
function* configuredModules(state) { | ||
var _a; | ||
for (let moduleConfig of INLINE_PRECOMPILE_MODULES) { | ||
if (moduleConfig.moduleName !== '@ember/template-compilation' && | ||
!((_a = state.opts.enableLegacyModules) === null || _a === void 0 ? void 0 : _a.includes(moduleConfig.moduleName))) { | ||
!state.normalizedOpts.enableLegacyModules.includes(moduleConfig.moduleName)) { | ||
continue; | ||
@@ -184,2 +155,134 @@ } | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plugin.js","sourceRoot":"","sources":["plugin.ts"],"names":[],"mappings":"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","sourcesContent":["import type { NodePath } from '@babel/traverse';\nimport type * as Babel from '@babel/core';\nimport type { types as t } from '@babel/core';\nimport { ImportUtil } from 'babel-import-util';\nimport { ExpressionParser } from './expression-parser';\nimport { JSUtils } from './js-utils';\n\nexport type EmberPrecompile = (templateString: string, options: Record<string, unknown>) => string;\n\nexport type LegacyModuleName =\n  | 'ember-cli-htmlbars'\n  | 'ember-cli-htmlbars-inline-precompile'\n  | 'htmlbars-inline-precompile';\n\ntype ModuleName = LegacyModuleName | '@ember/template-compilation';\n\ninterface ModuleConfig {\n  moduleName: ModuleName;\n  export: string;\n  allowTemplateLiteral: boolean;\n  enableScope: boolean;\n}\n\nconst INLINE_PRECOMPILE_MODULES: ModuleConfig[] = [\n  {\n    moduleName: 'ember-cli-htmlbars',\n    export: 'hbs',\n    allowTemplateLiteral: true,\n    enableScope: false,\n  },\n  {\n    moduleName: 'ember-cli-htmlbars-inline-precompile',\n    export: 'default',\n    allowTemplateLiteral: true,\n    enableScope: false,\n  },\n  {\n    moduleName: 'htmlbars-inline-precompile',\n    export: 'default',\n    allowTemplateLiteral: true,\n    enableScope: false,\n  },\n  {\n    moduleName: '@ember/template-compilation',\n    export: 'precompileTemplate',\n    allowTemplateLiteral: false,\n    enableScope: true,\n  },\n];\n\nexport interface Options {\n  // Allows you to remap what imports will be emitted in our compiled output. By\n  // example:\n  //\n  //   outputModuleOverrides: {\n  //     '@ember/template-factory': {\n  //       createTemplateFactory: ['createTemplateFactory', '@glimmer/core'],\n  //     }\n  //   }\n  //\n  // Normal Ember apps shouldn't need this, it exists to support other\n  // environments like standalone GlimmerJS\n  outputModuleOverrides?: Record<string, Record<string, [string, string]>>;\n\n  // By default, this plugin implements only Ember's stable public API for\n  // template compilation, which is:\n  //\n  //    import { precompileTemplate } from '@ember/template-compilation';\n  //\n  // But historically there are several other importable syntaxes in widespread\n  // use, and we can enable those too by including their module names in this\n  // list.\n  enableLegacyModules?: LegacyModuleName[];\n}\n\ninterface State {\n  opts: Options;\n  util: ImportUtil;\n  precompile: EmberPrecompile;\n  templateFactory: { moduleName: string; exportName: string };\n  program: NodePath<t.Program>;\n}\n\nexport default function makePlugin<O>(\n  // receives the Babel plugin options, returns Ember's precompiler\n  loadPrecompiler: (opts: O) => EmberPrecompile\n) {\n  return function htmlbarsInlinePrecompile(babel: typeof Babel): Babel.PluginObj<State> {\n    let t = babel.types;\n\n    function insertCompiledTemplate(\n      target: NodePath<t.Expression>,\n      state: State,\n      template: string,\n      userTypedOptions: Record<string, unknown>\n    ): void {\n      if (!userTypedOptions.locals) {\n        userTypedOptions.locals = [];\n      }\n      let jsutils = new JSUtils(\n        babel,\n        state.program,\n        target,\n        userTypedOptions.locals as string[],\n        state.util\n      );\n      let meta = Object.assign({ jsutils }, userTypedOptions?.meta);\n      let options = Object.assign({}, userTypedOptions, { contents: template, meta });\n      let precompile = state.precompile;\n      let precompileResultString: string;\n\n      if (options.insertRuntimeErrors) {\n        try {\n          precompileResultString = precompile(template, options);\n        } catch (error) {\n          target.replaceWith(runtimeErrorIIFE(babel, { ERROR_MESSAGE: (error as any).message }));\n          return;\n        }\n      } else {\n        precompileResultString = precompile(template, options);\n      }\n\n      let precompileResultAST = babel.parse(`var precompileResult = ${precompileResultString};`, {\n        babelrc: false,\n        configFile: false,\n      }) as t.File;\n\n      let templateExpression = (precompileResultAST.program.body[0] as t.VariableDeclaration)\n        .declarations[0].init as t.Expression;\n\n      t.addComment(\n        templateExpression,\n        'leading',\n        `\\n  ${template.replace(/\\*\\//g, '*\\\\/')}\\n`,\n        /* line comment? */ false\n      );\n\n      let templateFactoryIdentifier = state.util.import(\n        target,\n        state.templateFactory.moduleName,\n        state.templateFactory.exportName\n      );\n      target.replaceWith(t.callExpression(templateFactoryIdentifier, [templateExpression]));\n    }\n\n    return {\n      visitor: {\n        Program: {\n          enter(path: NodePath<t.Program>, state: State) {\n            let moduleName = '@ember/template-factory';\n            let exportName = 'createTemplateFactory';\n            let overrides = state.opts.outputModuleOverrides?.[moduleName]?.[exportName];\n            state.templateFactory = overrides\n              ? { exportName: overrides[0], moduleName: overrides[1] }\n              : { exportName, moduleName };\n            state.util = new ImportUtil(t, path);\n            state.precompile = loadPrecompiler(state.opts as O);\n            state.program = path;\n          },\n          exit(_path: NodePath<t.Program>, state: State) {\n            for (let { moduleName, export: exportName } of configuredModules(state)) {\n              state.util.removeImport(moduleName, exportName);\n            }\n          },\n        },\n\n        TaggedTemplateExpression(path: NodePath<t.TaggedTemplateExpression>, state: State) {\n          let tagPath = path.get('tag');\n\n          if (!tagPath.isIdentifier()) {\n            return;\n          }\n          let options = referencesInlineCompiler(tagPath, state);\n          if (!options) {\n            return;\n          }\n\n          if (!options.allowTemplateLiteral) {\n            throw path.buildCodeFrameError(\n              `Attempted to use \\`${tagPath.node.name}\\` as a template tag, but it can only be called as a function with a string passed to it: ${tagPath.node.name}('content here')`\n            );\n          }\n\n          if (path.node.quasi.expressions.length) {\n            throw path.buildCodeFrameError(\n              'placeholders inside a tagged template string are not supported'\n            );\n          }\n\n          let template = path.node.quasi.quasis.map((quasi) => quasi.value.cooked).join('');\n          insertCompiledTemplate(path, state, template, {});\n        },\n\n        CallExpression(path: NodePath<t.CallExpression>, state: State) {\n          let calleePath = path.get('callee');\n\n          if (!calleePath.isIdentifier()) {\n            return;\n          }\n          let options = referencesInlineCompiler(calleePath, state);\n          if (!options) {\n            return;\n          }\n\n          let [firstArg, secondArg, ...restArgs] = path.get('arguments');\n\n          let template;\n\n          switch (firstArg?.node.type) {\n            case 'StringLiteral':\n              template = firstArg.node.value;\n              break;\n            case 'TemplateLiteral':\n              if (firstArg.node.expressions.length) {\n                throw path.buildCodeFrameError(\n                  'placeholders inside a template string are not supported'\n                );\n              } else {\n                template = firstArg.node.quasis.map((quasi) => quasi.value.cooked).join('');\n              }\n              break;\n            case 'TaggedTemplateExpression':\n              throw path.buildCodeFrameError(\n                `tagged template strings inside ${calleePath.node.name} are not supported`\n              );\n            default:\n              throw path.buildCodeFrameError(\n                `${calleePath.node.name} should be invoked with at least a single argument (the template string)`\n              );\n          }\n\n          let userTypedOptions: Record<string, unknown>;\n\n          if (!secondArg) {\n            userTypedOptions = {};\n          } else {\n            if (!secondArg.isObjectExpression()) {\n              throw path.buildCodeFrameError(\n                `${calleePath.node.name} can only be invoked with 2 arguments: the template string, and any static options`\n              );\n            }\n\n            userTypedOptions = new ExpressionParser(babel).parseObjectExpression(\n              calleePath.node.name,\n              secondArg,\n              true\n            );\n          }\n          if (restArgs.length > 0) {\n            throw path.buildCodeFrameError(\n              `${calleePath.node.name} can only be invoked with 2 arguments: the template string, and any static options`\n            );\n          }\n          insertCompiledTemplate(path, state, template, userTypedOptions);\n        },\n      },\n    };\n  };\n}\n\nfunction* configuredModules(state: State) {\n  for (let moduleConfig of INLINE_PRECOMPILE_MODULES) {\n    if (\n      moduleConfig.moduleName !== '@ember/template-compilation' &&\n      !state.opts.enableLegacyModules?.includes(moduleConfig.moduleName)\n    ) {\n      continue;\n    }\n    yield moduleConfig;\n  }\n}\n\nfunction referencesInlineCompiler(\n  path: NodePath<t.Identifier>,\n  state: State\n): ModuleConfig | undefined {\n  for (let moduleConfig of configuredModules(state)) {\n    if (path.referencesImport(moduleConfig.moduleName, moduleConfig.export)) {\n      return moduleConfig;\n    }\n  }\n  return undefined;\n}\n\nfunction runtimeErrorIIFE(babel: typeof Babel, replacements: { ERROR_MESSAGE: string }) {\n  let statement = babel.template(`(function() {\\n  throw new Error('ERROR_MESSAGE');\\n})();`)(\n    replacements\n  ) as t.ExpressionStatement;\n  return statement.expression;\n}\n"]} | ||
function buildPrecompileOptions(babel, target, state, template, userTypedOptions) { | ||
if (!userTypedOptions.locals) { | ||
userTypedOptions.locals = []; | ||
} | ||
let jsutils = new js_utils_1.JSUtils(babel, state, target, userTypedOptions.locals, state.util); | ||
let meta = Object.assign({ jsutils }, userTypedOptions === null || userTypedOptions === void 0 ? void 0 : userTypedOptions.meta); | ||
return Object.assign({ | ||
contents: template, | ||
meta, | ||
// TODO: embroider's template-compiler allows this to be overriden to get | ||
// backward-compatible module names that don't match the real name of the | ||
// on-disk file. What's our plan for migrating people away from that? | ||
moduleName: state.filename, | ||
plugins: { | ||
ast: state.normalizedOpts.transforms, | ||
}, | ||
}, userTypedOptions); | ||
} | ||
function insertCompiledTemplate(babel, state, template, target, userTypedOptions) { | ||
let t = babel.types; | ||
let options = buildPrecompileOptions(babel, target, state, template, userTypedOptions); | ||
let precompileResultString; | ||
if (options.insertRuntimeErrors) { | ||
try { | ||
precompileResultString = state.normalizedOpts.compiler.precompile(template, options); | ||
} | ||
catch (error) { | ||
target.replaceWith(runtimeErrorIIFE(babel, { ERROR_MESSAGE: error.message })); | ||
return; | ||
} | ||
} | ||
else { | ||
precompileResultString = state.normalizedOpts.compiler.precompile(template, options); | ||
} | ||
let precompileResultAST = babel.parse(`var precompileResult = ${precompileResultString};`, { | ||
babelrc: false, | ||
configFile: false, | ||
}); | ||
let templateExpression = precompileResultAST.program.body[0] | ||
.declarations[0].init; | ||
t.addComment(templateExpression, 'leading', `\n ${template.replace(/\*\//g, '*\\/')}\n`, | ||
/* line comment? */ false); | ||
let templateFactoryIdentifier = state.util.import(target, state.templateFactory.moduleName, state.templateFactory.exportName); | ||
target.replaceWith(t.callExpression(templateFactoryIdentifier, [templateExpression])); | ||
} | ||
function insertTransformedTemplate(babel, state, template, target, userTypedOptions, formatOptions) { | ||
let t = babel.types; | ||
let options = buildPrecompileOptions(babel, target, state, template, userTypedOptions); | ||
let ast = state.normalizedOpts.compiler._preprocess(template, Object.assign(Object.assign({}, options), { mode: 'codemod' })); | ||
let transformed = state.normalizedOpts.compiler._print(ast); | ||
if (target.isCallExpression()) { | ||
target.get('arguments.0').replaceWith(t.stringLiteral(transformed)); | ||
if (options.locals && options.locals.length > 0) { | ||
if (!formatOptions.enableScope) { | ||
maybePruneImport(state.util, target.get('callee')); | ||
target.set('callee', precompileTemplate(state.util, target)); | ||
} | ||
updateScope(babel, target, options.locals); | ||
} | ||
} | ||
else { | ||
if (options.locals && options.locals.length > 0) { | ||
// need to add scope, so need to replace the backticks form with a call | ||
// expression to precompileTemplate | ||
maybePruneImport(state.util, target.get('tag')); | ||
let newCall = target.replaceWith(t.callExpression(precompileTemplate(state.util, target), [t.stringLiteral(transformed)]))[0]; | ||
updateScope(babel, newCall, options.locals); | ||
} | ||
else { | ||
target.get('quasi').get('quasis.0').replaceWith(t.templateElement({ raw: transformed })); | ||
} | ||
} | ||
} | ||
function templateFactoryConfig(opts) { | ||
var _a; | ||
let moduleName = '@ember/template-factory'; | ||
let exportName = 'createTemplateFactory'; | ||
let overrides = (_a = opts.outputModuleOverrides[moduleName]) === null || _a === void 0 ? void 0 : _a[exportName]; | ||
return overrides | ||
? { exportName: overrides[0], moduleName: overrides[1] } | ||
: { exportName, moduleName }; | ||
} | ||
function buildScope(babel, locals) { | ||
let t = babel.types; | ||
return t.arrowFunctionExpression([], t.objectExpression(locals.map((name) => t.objectProperty(t.identifier(name), t.identifier(name), false, true)))); | ||
} | ||
function updateScope(babel, target, locals) { | ||
let t = babel.types; | ||
let secondArg = target.get('arguments.1'); | ||
if (secondArg) { | ||
let scope = secondArg.get('properties').find((p) => { | ||
let key = p.get('key'); | ||
return key.isIdentifier() && key.node.name === 'scope'; | ||
}); | ||
if (scope) { | ||
scope.set('value', buildScope(babel, locals)); | ||
} | ||
else { | ||
secondArg.pushContainer('properties', t.objectProperty(t.identifier('scope'), buildScope(babel, locals))); | ||
} | ||
} | ||
else { | ||
target.pushContainer('arguments', t.objectExpression([t.objectProperty(t.identifier('scope'), buildScope(babel, locals))])); | ||
} | ||
} | ||
function maybePruneImport(util, identifier) { | ||
if (!identifier.isIdentifier()) { | ||
return; | ||
} | ||
let binding = identifier.scope.getBinding(identifier.node.name); | ||
// this checks if the identifier (that we're about to remove) is used in | ||
// exactly one place. | ||
if ((binding === null || binding === void 0 ? void 0 : binding.referencePaths.reduce((count, path) => (path.removed ? count : count + 1), 0)) === 1) { | ||
let specifier = binding.path; | ||
if (specifier.isImportSpecifier()) { | ||
let declaration = specifier.parentPath; | ||
util.removeImport(declaration.node.source.value, name(specifier.node.imported)); | ||
} | ||
} | ||
identifier.removed = true; | ||
} | ||
function precompileTemplate(util, target) { | ||
return util.import(target, '@ember/template-compilation', 'precompileTemplate'); | ||
} | ||
function name(node) { | ||
if (node.type === 'StringLiteral') { | ||
return node.value; | ||
} | ||
else { | ||
return node.name; | ||
} | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plugin.js","sourceRoot":"","sources":["plugin.ts"],"names":[],"mappings":";;AAGA,yDAA+C;AAC/C,2DAAuD;AACvD,yCAA4D;AAiB5D,MAAM,yBAAyB,GAAmB;IAChD;QACE,UAAU,EAAE,oBAAoB;QAChC,MAAM,EAAE,KAAK;QACb,oBAAoB,EAAE,IAAI;QAC1B,WAAW,EAAE,KAAK;KACnB;IACD;QACE,UAAU,EAAE,sCAAsC;QAClD,MAAM,EAAE,SAAS;QACjB,oBAAoB,EAAE,IAAI;QAC1B,WAAW,EAAE,KAAK;KACnB;IACD;QACE,UAAU,EAAE,4BAA4B;QACxC,MAAM,EAAE,SAAS;QACjB,oBAAoB,EAAE,IAAI;QAC1B,WAAW,EAAE,KAAK;KACnB;IACD;QACE,UAAU,EAAE,6BAA6B;QACzC,MAAM,EAAE,oBAAoB;QAC5B,oBAAoB,EAAE,KAAK;QAC3B,WAAW,EAAE,IAAI;KAClB;CACF,CAAC;AAuDF,SAAwB,UAAU,CAChC,WAAkD;IAElD,OAAO,SAAS,wBAAwB,CACtC,KAAmB;QAEnB,IAAI,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC;QAEpB,OAAO;YACL,OAAO,EAAE;gBACP,OAAO,EAAE;oBACP,KAAK,CAAC,IAAyB,EAAE,KAAgC;wBAC/D,KAAK,CAAC,cAAc,mBAClB,YAAY,EAAE,MAAM,EACpB,qBAAqB,EAAE,EAAE,EACzB,mBAAmB,EAAE,EAAE,EACvB,UAAU,EAAE,EAAE,IACX,WAAW,CAAC,KAAK,CAAC,IAAI,CAAC,CAC3B,CAAC;wBAEF,KAAK,CAAC,eAAe,GAAG,qBAAqB,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;wBACpE,KAAK,CAAC,IAAI,GAAG,IAAI,8BAAU,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC;wBACrC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC;oBACvB,CAAC;oBACD,IAAI,CAAC,KAA0B,EAAE,KAAgC;wBAC/D,IAAI,KAAK,CAAC,cAAc,CAAC,YAAY,KAAK,MAAM,EAAE;4BAChD,KAAK,IAAI,EAAE,UAAU,EAAE,MAAM,EAAE,UAAU,EAAE,IAAI,iBAAiB,CAAC,KAAK,CAAC,EAAE;gCACvE,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,UAAU,EAAE,UAAU,CAAC,CAAC;6BACjD;yBACF;oBACH,CAAC;iBACF;gBAED,wBAAwB,CACtB,IAA0C,EAC1C,KAAgC;oBAEhC,IAAI,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;oBAE9B,IAAI,CAAC,OAAO,CAAC,YAAY,EAAE,EAAE;wBAC3B,OAAO;qBACR;oBACD,IAAI,OAAO,GAAG,wBAAwB,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;oBACvD,IAAI,CAAC,OAAO,EAAE;wBACZ,OAAO;qBACR;oBAED,IAAI,CAAC,OAAO,CAAC,oBAAoB,EAAE;wBACjC,MAAM,IAAI,CAAC,mBAAmB,CAC5B,sBAAsB,OAAO,CAAC,IAAI,CAAC,IAAI,6FAA6F,OAAO,CAAC,IAAI,CAAC,IAAI,kBAAkB,CACxK,CAAC;qBACH;oBAED,IAAI,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,MAAM,EAAE;wBACtC,MAAM,IAAI,CAAC,mBAAmB,CAC5B,gEAAgE,CACjE,CAAC;qBACH;oBAED,IAAI,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;oBAClF,IAAI,KAAK,CAAC,cAAc,CAAC,YAAY,KAAK,MAAM,EAAE;wBAChD,sBAAsB,CAAC,KAAK,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,EAAE,CAAC,CAAC;qBAC1D;yBAAM;wBACL,yBAAyB,CAAC,KAAK,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,EAAE,EAAE,OAAO,CAAC,CAAC;qBACtE;gBACH,CAAC;gBAED,cAAc,CAAC,IAAgC,EAAE,KAAgC;oBAC/E,IAAI,UAAU,GAAG,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;oBAEpC,IAAI,CAAC,UAAU,CAAC,YAAY,EAAE,EAAE;wBAC9B,OAAO;qBACR;oBACD,IAAI,OAAO,GAAG,wBAAwB,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC;oBAC1D,IAAI,CAAC,OAAO,EAAE;wBACZ,OAAO;qBACR;oBAED,IAAI,CAAC,QAAQ,EAAE,SAAS,EAAE,GAAG,QAAQ,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;oBAE/D,IAAI,QAAQ,CAAC;oBAEb,QAAQ,QAAQ,aAAR,QAAQ,uBAAR,QAAQ,CAAE,IAAI,CAAC,IAAI,EAAE;wBAC3B,KAAK,eAAe;4BAClB,QAAQ,GAAG,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC;4BAC/B,MAAM;wBACR,KAAK,iBAAiB;4BACpB,IAAI,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,MAAM,EAAE;gCACpC,MAAM,IAAI,CAAC,mBAAmB,CAC5B,yDAAyD,CAC1D,CAAC;6BACH;iCAAM;gCACL,QAAQ,GAAG,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;6BAC7E;4BACD,MAAM;wBACR,KAAK,0BAA0B;4BAC7B,MAAM,IAAI,CAAC,mBAAmB,CAC5B,kCAAkC,UAAU,CAAC,IAAI,CAAC,IAAI,oBAAoB,CAC3E,CAAC;wBACJ;4BACE,MAAM,IAAI,CAAC,mBAAmB,CAC5B,GAAG,UAAU,CAAC,IAAI,CAAC,IAAI,0EAA0E,CAClG,CAAC;qBACL;oBAED,IAAI,gBAAyC,CAAC;oBAE9C,IAAI,CAAC,SAAS,EAAE;wBACd,gBAAgB,GAAG,EAAE,CAAC;qBACvB;yBAAM;wBACL,IAAI,CAAC,SAAS,CAAC,kBAAkB,EAAE,EAAE;4BACnC,MAAM,IAAI,CAAC,mBAAmB,CAC5B,GAAG,UAAU,CAAC,IAAI,CAAC,IAAI,oFAAoF,CAC5G,CAAC;yBACH;wBAED,gBAAgB,GAAG,IAAI,oCAAgB,CAAC,KAAK,CAAC,CAAC,qBAAqB,CAClE,UAAU,CAAC,IAAI,CAAC,IAAI,EACpB,SAAS,EACT,OAAO,CAAC,WAAW,CACpB,CAAC;qBACH;oBACD,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE;wBACvB,MAAM,IAAI,CAAC,mBAAmB,CAC5B,GAAG,UAAU,CAAC,IAAI,CAAC,IAAI,oFAAoF,CAC5G,CAAC;qBACH;oBACD,IAAI,KAAK,CAAC,cAAc,CAAC,YAAY,KAAK,MAAM,EAAE;wBAChD,sBAAsB,CAAC,KAAK,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,gBAAgB,CAAC,CAAC;qBACxE;yBAAM;wBACL,yBAAyB,CAAC,KAAK,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,gBAAgB,EAAE,OAAO,CAAC,CAAC;qBACpF;gBACH,CAAC;aACF;SACF,CAAC;IACJ,CAAC,CAAC;AACJ,CAAC;AAxID,6BAwIC;AAED,QAAQ,CAAC,CAAC,iBAAiB,CAAqB,KAAgC;IAC9E,KAAK,IAAI,YAAY,IAAI,yBAAyB,EAAE;QAClD,IACE,YAAY,CAAC,UAAU,KAAK,6BAA6B;YACzD,CAAC,KAAK,CAAC,cAAc,CAAC,mBAAmB,CAAC,QAAQ,CAAC,YAAY,CAAC,UAAU,CAAC,EAC3E;YACA,SAAS;SACV;QACD,MAAM,YAAY,CAAC;KACpB;AACH,CAAC;AAED,SAAS,wBAAwB,CAC/B,IAA4B,EAC5B,KAAgC;IAEhC,KAAK,IAAI,YAAY,IAAI,iBAAiB,CAAC,KAAK,CAAC,EAAE;QACjD,IAAI,IAAI,CAAC,gBAAgB,CAAC,YAAY,CAAC,UAAU,EAAE,YAAY,CAAC,MAAM,CAAC,EAAE;YACvE,OAAO,YAAY,CAAC;SACrB;KACF;IACD,OAAO,SAAS,CAAC;AACnB,CAAC;AAED,SAAS,gBAAgB,CAAC,KAAmB,EAAE,YAAuC;IACpF,IAAI,SAAS,GAAG,KAAK,CAAC,QAAQ,CAAC,2DAA2D,CAAC,CACzF,YAAY,CACY,CAAC;IAC3B,OAAO,SAAS,CAAC,UAAU,CAAC;AAC9B,CAAC;AAED,SAAS,sBAAsB,CAC7B,KAAmB,EACnB,MAA8B,EAC9B,KAAgC,EAChC,QAAgB,EAChB,gBAAyC;IAEzC,IAAI,CAAC,gBAAgB,CAAC,MAAM,EAAE;QAC5B,gBAAgB,CAAC,MAAM,GAAG,EAAE,CAAC;KAC9B;IACD,IAAI,OAAO,GAAG,IAAI,kBAAO,CAAC,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,gBAAgB,CAAC,MAAkB,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;IACjG,IAAI,IAAI,GAAG,MAAM,CAAC,MAAM,CAAC,EAAE,OAAO,EAAE,EAAE,gBAAgB,aAAhB,gBAAgB,uBAAhB,gBAAgB,CAAE,IAAI,CAAC,CAAC;IAC9D,OAAO,MAAM,CAAC,MAAM,CAClB;QACE,QAAQ,EAAE,QAAQ;QAClB,IAAI;QAEJ,yEAAyE;QACzE,yEAAyE;QACzE,qEAAqE;QACrE,UAAU,EAAE,KAAK,CAAC,QAAQ;QAE1B,OAAO,EAAE;YACP,GAAG,EAAE,KAAK,CAAC,cAAc,CAAC,UAAU;SACrC;KACF,EACD,gBAAgB,CACjB,CAAC;AACJ,CAAC;AAED,SAAS,sBAAsB,CAC7B,KAAmB,EACnB,KAAgC,EAChC,QAAgB,EAChB,MAA8B,EAC9B,gBAAyC;IAEzC,IAAI,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC;IACpB,IAAI,OAAO,GAAG,sBAAsB,CAAC,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,QAAQ,EAAE,gBAAgB,CAAC,CAAC;IAEvF,IAAI,sBAA8B,CAAC;IAEnC,IAAI,OAAO,CAAC,mBAAmB,EAAE;QAC/B,IAAI;YACF,sBAAsB,GAAG,KAAK,CAAC,cAAc,CAAC,QAAQ,CAAC,UAAU,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;SACtF;QAAC,OAAO,KAAK,EAAE;YACd,MAAM,CAAC,WAAW,CAAC,gBAAgB,CAAC,KAAK,EAAE,EAAE,aAAa,EAAG,KAAa,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;YACvF,OAAO;SACR;KACF;SAAM;QACL,sBAAsB,GAAG,KAAK,CAAC,cAAc,CAAC,QAAQ,CAAC,UAAU,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;KACtF;IAED,IAAI,mBAAmB,GAAG,KAAK,CAAC,KAAK,CAAC,0BAA0B,sBAAsB,GAAG,EAAE;QACzF,OAAO,EAAE,KAAK;QACd,UAAU,EAAE,KAAK;KAClB,CAAW,CAAC;IAEb,IAAI,kBAAkB,GAAI,mBAAmB,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAA2B;SACpF,YAAY,CAAC,CAAC,CAAC,CAAC,IAAoB,CAAC;IAExC,CAAC,CAAC,UAAU,CACV,kBAAkB,EAClB,SAAS,EACT,OAAO,QAAQ,CAAC,OAAO,CAAC,OAAO,EAAE,MAAM,CAAC,IAAI;IAC5C,mBAAmB,CAAC,KAAK,CAC1B,CAAC;IAEF,IAAI,yBAAyB,GAAG,KAAK,CAAC,IAAI,CAAC,MAAM,CAC/C,MAAM,EACN,KAAK,CAAC,eAAe,CAAC,UAAU,EAChC,KAAK,CAAC,eAAe,CAAC,UAAU,CACjC,CAAC;IACF,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,cAAc,CAAC,yBAAyB,EAAE,CAAC,kBAAkB,CAAC,CAAC,CAAC,CAAC;AACxF,CAAC;AAED,SAAS,yBAAyB,CAChC,KAAmB,EACnB,KAAgC,EAChC,QAAgB,EAChB,MAAyE,EACzE,gBAAyC,EACzC,aAA2B;IAE3B,IAAI,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC;IACpB,IAAI,OAAO,GAAG,sBAAsB,CAAC,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,QAAQ,EAAE,gBAAgB,CAAC,CAAC;IACvF,IAAI,GAAG,GAAG,KAAK,CAAC,cAAc,CAAC,QAAQ,CAAC,WAAW,CAAC,QAAQ,kCAAO,OAAO,KAAE,IAAI,EAAE,SAAS,IAAG,CAAC;IAC/F,IAAI,WAAW,GAAG,KAAK,CAAC,cAAc,CAAC,QAAQ,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;IAC5D,IAAI,MAAM,CAAC,gBAAgB,EAAE,EAAE;QAC5B,MAAM,CAAC,GAAG,CAAC,aAAa,CAAsB,CAAC,WAAW,CAAC,CAAC,CAAC,aAAa,CAAC,WAAW,CAAC,CAAC,CAAC;QAC1F,IAAI,OAAO,CAAC,MAAM,IAAI,OAAO,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE;YAC/C,IAAI,CAAC,aAAa,CAAC,WAAW,EAAE;gBAC9B,gBAAgB,CAAC,KAAK,CAAC,IAAI,EAAE,MAAM,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC,CAAC;gBACnD,MAAM,CAAC,GAAG,CAAC,QAAQ,EAAE,kBAAkB,CAAC,KAAK,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC,CAAC;aAC9D;YACD,WAAW,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC;SAC5C;KACF;SAAM;QACL,IAAI,OAAO,CAAC,MAAM,IAAI,OAAO,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE;YAC/C,uEAAuE;YACvE,mCAAmC;YACnC,gBAAgB,CAAC,KAAK,CAAC,IAAI,EAAE,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC;YAChD,IAAI,OAAO,GAAG,MAAM,CAAC,WAAW,CAC9B,CAAC,CAAC,cAAc,CAAC,kBAAkB,CAAC,KAAK,CAAC,IAAI,EAAE,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,aAAa,CAAC,WAAW,CAAC,CAAC,CAAC,CACzF,CAAC,CAAC,CAAC,CAAC;YACL,WAAW,CAAC,KAAK,EAAE,OAAO,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC;SAC7C;aAAM;YACJ,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,GAAG,CAAC,UAAU,CAAiC,CAAC,WAAW,CAC9E,CAAC,CAAC,eAAe,CAAC,EAAE,GAAG,EAAE,WAAW,EAAE,CAAC,CACxC,CAAC;SACH;KACF;AACH,CAAC;AAED,SAAS,qBAAqB,CAAC,IAAuB;;IACpD,IAAI,UAAU,GAAG,yBAAyB,CAAC;IAC3C,IAAI,UAAU,GAAG,uBAAuB,CAAC;IACzC,IAAI,SAAS,GAAG,MAAA,IAAI,CAAC,qBAAqB,CAAC,UAAU,CAAC,0CAAG,UAAU,CAAC,CAAC;IACrE,OAAO,SAAS;QACd,CAAC,CAAC,EAAE,UAAU,EAAE,SAAS,CAAC,CAAC,CAAC,EAAE,UAAU,EAAE,SAAS,CAAC,CAAC,CAAC,EAAE;QACxD,CAAC,CAAC,EAAE,UAAU,EAAE,UAAU,EAAE,CAAC;AACjC,CAAC;AAED,SAAS,UAAU,CAAC,KAAmB,EAAE,MAAgB;IACvD,IAAI,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC;IACpB,OAAO,CAAC,CAAC,uBAAuB,CAC9B,EAAE,EACF,CAAC,CAAC,gBAAgB,CAChB,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC,CAC5F,CACF,CAAC;AACJ,CAAC;AACD,SAAS,WAAW,CAAC,KAAmB,EAAE,MAAkC,EAAE,MAAgB;IAC5F,IAAI,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC;IACpB,IAAI,SAAS,GAAG,MAAM,CAAC,GAAG,CAAC,aAAa,CAA6C,CAAC;IACtF,IAAI,SAAS,EAAE;QACb,IAAI,KAAK,GAAG,SAAS,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE;YACjD,IAAI,GAAG,GAAG,CAAC,CAAC,GAAG,CAAC,KAAK,CAAqB,CAAC;YAC3C,OAAO,GAAG,CAAC,YAAY,EAAE,IAAI,GAAG,CAAC,IAAI,CAAC,IAAI,KAAK,OAAO,CAAC;QACzD,CAAC,CAAC,CAAC;QACH,IAAI,KAAK,EAAE;YACT,KAAK,CAAC,GAAG,CAAC,OAAO,EAAE,UAAU,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC,CAAC;SAC/C;aAAM;YACL,SAAS,CAAC,aAAa,CACrB,YAAY,EACZ,CAAC,CAAC,cAAc,CAAC,CAAC,CAAC,UAAU,CAAC,OAAO,CAAC,EAAE,UAAU,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC,CACnE,CAAC;SACH;KACF;SAAM;QACL,MAAM,CAAC,aAAa,CAClB,WAAW,EACX,CAAC,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC,CAAC,UAAU,CAAC,OAAO,CAAC,EAAE,UAAU,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC,CACzF,CAAC;KACH;AACH,CAAC;AAED,SAAS,gBAAgB,CACvB,IAAgB,EAChB,UAA4D;IAE5D,IAAI,CAAC,UAAU,CAAC,YAAY,EAAE,EAAE;QAC9B,OAAO;KACR;IACD,IAAI,OAAO,GAAG,UAAU,CAAC,KAAK,CAAC,UAAU,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAChE,wEAAwE;IACxE,qBAAqB;IACrB,IACE,CAAA,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,cAAc,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC,MAAK,CAAC,EAC5F;QACA,IAAI,SAAS,GAAG,OAAO,CAAC,IAAI,CAAC;QAC7B,IAAI,SAAS,CAAC,iBAAiB,EAAE,EAAE;YACjC,IAAI,WAAW,GAAG,SAAS,CAAC,UAA2C,CAAC;YACxE,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;SACjF;KACF;IACD,UAAU,CAAC,OAAO,GAAG,IAAI,CAAC;AAC5B,CAAC;AAED,SAAS,kBAAkB,CAAC,IAAgB,EAAE,MAAwB;IACpE,OAAO,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,6BAA6B,EAAE,oBAAoB,CAAC,CAAC;AAClF,CAAC;AAED,SAAS,IAAI,CAAC,IAAoC;IAChD,IAAI,IAAI,CAAC,IAAI,KAAK,eAAe,EAAE;QACjC,OAAO,IAAI,CAAC,KAAK,CAAC;KACnB;SAAM;QACL,OAAO,IAAI,CAAC,IAAI,CAAC;KAClB;AACH,CAAC","sourcesContent":["import type { NodePath } from '@babel/traverse';\nimport type * as Babel from '@babel/core';\nimport type { types as t } from '@babel/core';\nimport { ImportUtil } from 'babel-import-util';\nimport { ExpressionParser } from './expression-parser';\nimport { JSUtils, ExtendedPluginBuilder } from './js-utils';\nimport type { EmberTemplateCompiler, PreprocessOptions } from './ember-template-compiler';\n\nexport type LegacyModuleName =\n  | 'ember-cli-htmlbars'\n  | 'ember-cli-htmlbars-inline-precompile'\n  | 'htmlbars-inline-precompile';\n\ntype ModuleName = LegacyModuleName | '@ember/template-compilation';\n\ninterface ModuleConfig {\n  moduleName: ModuleName;\n  export: string;\n  allowTemplateLiteral: boolean;\n  enableScope: boolean;\n}\n\nconst INLINE_PRECOMPILE_MODULES: ModuleConfig[] = [\n  {\n    moduleName: 'ember-cli-htmlbars',\n    export: 'hbs',\n    allowTemplateLiteral: true,\n    enableScope: false,\n  },\n  {\n    moduleName: 'ember-cli-htmlbars-inline-precompile',\n    export: 'default',\n    allowTemplateLiteral: true,\n    enableScope: false,\n  },\n  {\n    moduleName: 'htmlbars-inline-precompile',\n    export: 'default',\n    allowTemplateLiteral: true,\n    enableScope: false,\n  },\n  {\n    moduleName: '@ember/template-compilation',\n    export: 'precompileTemplate',\n    allowTemplateLiteral: false,\n    enableScope: true,\n  },\n];\n\nexport interface Options {\n  // The ember-template-compiler.js module that ships within your ember-source version.\n  compiler: EmberTemplateCompiler;\n\n  // Allows you to remap what imports will be emitted in our compiled output. By\n  // example:\n  //\n  //   outputModuleOverrides: {\n  //     '@ember/template-factory': {\n  //       createTemplateFactory: ['createTemplateFactory', '@glimmer/core'],\n  //     }\n  //   }\n  //\n  // Normal Ember apps shouldn't need this, it exists to support other\n  // environments like standalone GlimmerJS\n  outputModuleOverrides?: Record<string, Record<string, [string, string]>>;\n\n  // By default, this plugin implements only Ember's stable public API for\n  // template compilation, which is:\n  //\n  //    import { precompileTemplate } from '@ember/template-compilation';\n  //\n  // But historically there are several other importable syntaxes in widespread\n  // use, and we can enable those too by including their module names in this\n  // list.\n  enableLegacyModules?: LegacyModuleName[];\n\n  // Controls the output format.\n  //\n  //  \"wire\": The default. In the output, your templates are ready to execute in\n  //  the most performant way.\n  //\n  //  \"hbs\": In the output, your templates will still be in HBS format.\n  //  Generally this means they will still need further processing before\n  //  they're ready to execute. The purpose of this mode is to support things\n  //  like codemods and pre-publication transformations in libraries.\n  targetFormat?: 'wire' | 'hbs';\n\n  // Optional list of custom transforms to apply to the handlebars AST before\n  // compilation.\n  transforms?: ExtendedPluginBuilder[];\n}\n\nexport interface State<EnvSpecificOptions> {\n  opts: EnvSpecificOptions;\n  normalizedOpts: Required<Options>;\n  util: ImportUtil;\n  templateFactory: { moduleName: string; exportName: string };\n  program: NodePath<t.Program>;\n  lastInsertedPath: NodePath<t.Statement> | undefined;\n  filename: string;\n}\n\nexport default function makePlugin<EnvSpecificOptions>(\n  loadOptions: (opts: EnvSpecificOptions) => Options\n) {\n  return function htmlbarsInlinePrecompile(\n    babel: typeof Babel\n  ): Babel.PluginObj<State<EnvSpecificOptions>> {\n    let t = babel.types;\n\n    return {\n      visitor: {\n        Program: {\n          enter(path: NodePath<t.Program>, state: State<EnvSpecificOptions>) {\n            state.normalizedOpts = {\n              targetFormat: 'wire',\n              outputModuleOverrides: {},\n              enableLegacyModules: [],\n              transforms: [],\n              ...loadOptions(state.opts),\n            };\n\n            state.templateFactory = templateFactoryConfig(state.normalizedOpts);\n            state.util = new ImportUtil(t, path);\n            state.program = path;\n          },\n          exit(_path: NodePath<t.Program>, state: State<EnvSpecificOptions>) {\n            if (state.normalizedOpts.targetFormat === 'wire') {\n              for (let { moduleName, export: exportName } of configuredModules(state)) {\n                state.util.removeImport(moduleName, exportName);\n              }\n            }\n          },\n        },\n\n        TaggedTemplateExpression(\n          path: NodePath<t.TaggedTemplateExpression>,\n          state: State<EnvSpecificOptions>\n        ) {\n          let tagPath = path.get('tag');\n\n          if (!tagPath.isIdentifier()) {\n            return;\n          }\n          let options = referencesInlineCompiler(tagPath, state);\n          if (!options) {\n            return;\n          }\n\n          if (!options.allowTemplateLiteral) {\n            throw path.buildCodeFrameError(\n              `Attempted to use \\`${tagPath.node.name}\\` as a template tag, but it can only be called as a function with a string passed to it: ${tagPath.node.name}('content here')`\n            );\n          }\n\n          if (path.node.quasi.expressions.length) {\n            throw path.buildCodeFrameError(\n              'placeholders inside a tagged template string are not supported'\n            );\n          }\n\n          let template = path.node.quasi.quasis.map((quasi) => quasi.value.cooked).join('');\n          if (state.normalizedOpts.targetFormat === 'wire') {\n            insertCompiledTemplate(babel, state, template, path, {});\n          } else {\n            insertTransformedTemplate(babel, state, template, path, {}, options);\n          }\n        },\n\n        CallExpression(path: NodePath<t.CallExpression>, state: State<EnvSpecificOptions>) {\n          let calleePath = path.get('callee');\n\n          if (!calleePath.isIdentifier()) {\n            return;\n          }\n          let options = referencesInlineCompiler(calleePath, state);\n          if (!options) {\n            return;\n          }\n\n          let [firstArg, secondArg, ...restArgs] = path.get('arguments');\n\n          let template;\n\n          switch (firstArg?.node.type) {\n            case 'StringLiteral':\n              template = firstArg.node.value;\n              break;\n            case 'TemplateLiteral':\n              if (firstArg.node.expressions.length) {\n                throw path.buildCodeFrameError(\n                  'placeholders inside a template string are not supported'\n                );\n              } else {\n                template = firstArg.node.quasis.map((quasi) => quasi.value.cooked).join('');\n              }\n              break;\n            case 'TaggedTemplateExpression':\n              throw path.buildCodeFrameError(\n                `tagged template strings inside ${calleePath.node.name} are not supported`\n              );\n            default:\n              throw path.buildCodeFrameError(\n                `${calleePath.node.name} should be invoked with at least a single argument (the template string)`\n              );\n          }\n\n          let userTypedOptions: Record<string, unknown>;\n\n          if (!secondArg) {\n            userTypedOptions = {};\n          } else {\n            if (!secondArg.isObjectExpression()) {\n              throw path.buildCodeFrameError(\n                `${calleePath.node.name} can only be invoked with 2 arguments: the template string, and any static options`\n              );\n            }\n\n            userTypedOptions = new ExpressionParser(babel).parseObjectExpression(\n              calleePath.node.name,\n              secondArg,\n              options.enableScope\n            );\n          }\n          if (restArgs.length > 0) {\n            throw path.buildCodeFrameError(\n              `${calleePath.node.name} can only be invoked with 2 arguments: the template string, and any static options`\n            );\n          }\n          if (state.normalizedOpts.targetFormat === 'wire') {\n            insertCompiledTemplate(babel, state, template, path, userTypedOptions);\n          } else {\n            insertTransformedTemplate(babel, state, template, path, userTypedOptions, options);\n          }\n        },\n      },\n    };\n  };\n}\n\nfunction* configuredModules<EnvSpecificOptions>(state: State<EnvSpecificOptions>) {\n  for (let moduleConfig of INLINE_PRECOMPILE_MODULES) {\n    if (\n      moduleConfig.moduleName !== '@ember/template-compilation' &&\n      !state.normalizedOpts.enableLegacyModules.includes(moduleConfig.moduleName)\n    ) {\n      continue;\n    }\n    yield moduleConfig;\n  }\n}\n\nfunction referencesInlineCompiler<EnvSpecificOptions>(\n  path: NodePath<t.Identifier>,\n  state: State<EnvSpecificOptions>\n): ModuleConfig | undefined {\n  for (let moduleConfig of configuredModules(state)) {\n    if (path.referencesImport(moduleConfig.moduleName, moduleConfig.export)) {\n      return moduleConfig;\n    }\n  }\n  return undefined;\n}\n\nfunction runtimeErrorIIFE(babel: typeof Babel, replacements: { ERROR_MESSAGE: string }) {\n  let statement = babel.template(`(function() {\\n  throw new Error('ERROR_MESSAGE');\\n})();`)(\n    replacements\n  ) as t.ExpressionStatement;\n  return statement.expression;\n}\n\nfunction buildPrecompileOptions<EnvSpecificOptions>(\n  babel: typeof Babel,\n  target: NodePath<t.Expression>,\n  state: State<EnvSpecificOptions>,\n  template: string,\n  userTypedOptions: Record<string, unknown>\n): PreprocessOptions & Record<string, unknown> {\n  if (!userTypedOptions.locals) {\n    userTypedOptions.locals = [];\n  }\n  let jsutils = new JSUtils(babel, state, target, userTypedOptions.locals as string[], state.util);\n  let meta = Object.assign({ jsutils }, userTypedOptions?.meta);\n  return Object.assign(\n    {\n      contents: template,\n      meta,\n\n      // TODO: embroider's template-compiler allows this to be overriden to get\n      // backward-compatible module names that don't match the real name of the\n      // on-disk file. What's our plan for migrating people away from that?\n      moduleName: state.filename,\n\n      plugins: {\n        ast: state.normalizedOpts.transforms,\n      },\n    },\n    userTypedOptions\n  );\n}\n\nfunction insertCompiledTemplate<EnvSpecificOptions>(\n  babel: typeof Babel,\n  state: State<EnvSpecificOptions>,\n  template: string,\n  target: NodePath<t.Expression>,\n  userTypedOptions: Record<string, unknown>\n) {\n  let t = babel.types;\n  let options = buildPrecompileOptions(babel, target, state, template, userTypedOptions);\n\n  let precompileResultString: string;\n\n  if (options.insertRuntimeErrors) {\n    try {\n      precompileResultString = state.normalizedOpts.compiler.precompile(template, options);\n    } catch (error) {\n      target.replaceWith(runtimeErrorIIFE(babel, { ERROR_MESSAGE: (error as any).message }));\n      return;\n    }\n  } else {\n    precompileResultString = state.normalizedOpts.compiler.precompile(template, options);\n  }\n\n  let precompileResultAST = babel.parse(`var precompileResult = ${precompileResultString};`, {\n    babelrc: false,\n    configFile: false,\n  }) as t.File;\n\n  let templateExpression = (precompileResultAST.program.body[0] as t.VariableDeclaration)\n    .declarations[0].init as t.Expression;\n\n  t.addComment(\n    templateExpression,\n    'leading',\n    `\\n  ${template.replace(/\\*\\//g, '*\\\\/')}\\n`,\n    /* line comment? */ false\n  );\n\n  let templateFactoryIdentifier = state.util.import(\n    target,\n    state.templateFactory.moduleName,\n    state.templateFactory.exportName\n  );\n  target.replaceWith(t.callExpression(templateFactoryIdentifier, [templateExpression]));\n}\n\nfunction insertTransformedTemplate<EnvSpecificOptions>(\n  babel: typeof Babel,\n  state: State<EnvSpecificOptions>,\n  template: string,\n  target: NodePath<t.CallExpression> | NodePath<t.TaggedTemplateExpression>,\n  userTypedOptions: Record<string, unknown>,\n  formatOptions: ModuleConfig\n) {\n  let t = babel.types;\n  let options = buildPrecompileOptions(babel, target, state, template, userTypedOptions);\n  let ast = state.normalizedOpts.compiler._preprocess(template, { ...options, mode: 'codemod' });\n  let transformed = state.normalizedOpts.compiler._print(ast);\n  if (target.isCallExpression()) {\n    (target.get('arguments.0') as NodePath<t.Node>).replaceWith(t.stringLiteral(transformed));\n    if (options.locals && options.locals.length > 0) {\n      if (!formatOptions.enableScope) {\n        maybePruneImport(state.util, target.get('callee'));\n        target.set('callee', precompileTemplate(state.util, target));\n      }\n      updateScope(babel, target, options.locals);\n    }\n  } else {\n    if (options.locals && options.locals.length > 0) {\n      // need to add scope, so need to replace the backticks form with a call\n      // expression to precompileTemplate\n      maybePruneImport(state.util, target.get('tag'));\n      let newCall = target.replaceWith(\n        t.callExpression(precompileTemplate(state.util, target), [t.stringLiteral(transformed)])\n      )[0];\n      updateScope(babel, newCall, options.locals);\n    } else {\n      (target.get('quasi').get('quasis.0') as NodePath<t.TemplateElement>).replaceWith(\n        t.templateElement({ raw: transformed })\n      );\n    }\n  }\n}\n\nfunction templateFactoryConfig(opts: Required<Options>) {\n  let moduleName = '@ember/template-factory';\n  let exportName = 'createTemplateFactory';\n  let overrides = opts.outputModuleOverrides[moduleName]?.[exportName];\n  return overrides\n    ? { exportName: overrides[0], moduleName: overrides[1] }\n    : { exportName, moduleName };\n}\n\nfunction buildScope(babel: typeof Babel, locals: string[]) {\n  let t = babel.types;\n  return t.arrowFunctionExpression(\n    [],\n    t.objectExpression(\n      locals.map((name) => t.objectProperty(t.identifier(name), t.identifier(name), false, true))\n    )\n  );\n}\nfunction updateScope(babel: typeof Babel, target: NodePath<t.CallExpression>, locals: string[]) {\n  let t = babel.types;\n  let secondArg = target.get('arguments.1') as NodePath<t.ObjectExpression> | undefined;\n  if (secondArg) {\n    let scope = secondArg.get('properties').find((p) => {\n      let key = p.get('key') as NodePath<t.Node>;\n      return key.isIdentifier() && key.node.name === 'scope';\n    });\n    if (scope) {\n      scope.set('value', buildScope(babel, locals));\n    } else {\n      secondArg.pushContainer(\n        'properties',\n        t.objectProperty(t.identifier('scope'), buildScope(babel, locals))\n      );\n    }\n  } else {\n    target.pushContainer(\n      'arguments',\n      t.objectExpression([t.objectProperty(t.identifier('scope'), buildScope(babel, locals))])\n    );\n  }\n}\n\nfunction maybePruneImport(\n  util: ImportUtil,\n  identifier: NodePath<t.Expression | t.V8IntrinsicIdentifier>\n) {\n  if (!identifier.isIdentifier()) {\n    return;\n  }\n  let binding = identifier.scope.getBinding(identifier.node.name);\n  // this checks if the identifier (that we're about to remove) is used in\n  // exactly one place.\n  if (\n    binding?.referencePaths.reduce((count, path) => (path.removed ? count : count + 1), 0) === 1\n  ) {\n    let specifier = binding.path;\n    if (specifier.isImportSpecifier()) {\n      let declaration = specifier.parentPath as NodePath<t.ImportDeclaration>;\n      util.removeImport(declaration.node.source.value, name(specifier.node.imported));\n    }\n  }\n  identifier.removed = true;\n}\n\nfunction precompileTemplate(util: ImportUtil, target: NodePath<t.Node>) {\n  return util.import(target, '@ember/template-compilation', 'precompileTemplate');\n}\n\nfunction name(node: t.StringLiteral | t.Identifier) {\n  if (node.type === 'StringLiteral') {\n    return node.value;\n  } else {\n    return node.name;\n  }\n}\n"]} |
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
Dynamic require
Supply chain riskDynamic require can indicate the package is performing dangerous or unsafe dynamic code execution.
Found 1 instance in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
Dynamic require
Supply chain riskDynamic require can indicate the package is performing dangerous or unsafe dynamic code execution.
Found 1 instance in 1 package
151390
17
915
24
3