Socket
Socket
Sign inDemoInstall

babel-plugin-ember-template-compilation

Package Overview
Dependencies
8
Maintainers
17
Versions
22
Alerts
File Explorer

Advanced tools

Install Socket

Detect and block malicious and high-risk dependencies

Install

Comparing version 2.2.1 to 2.2.2

src/hbs-utils.d.ts

2

package.json
{
"name": "babel-plugin-ember-template-compilation",
"version": "2.2.1",
"version": "2.2.2",
"description": "Babel implementation of Ember's low-level template-compilation API",

@@ -5,0 +5,0 @@ "repository": "https://github.com/emberjs/babel-plugin-ember-template-compilation",

@@ -80,3 +80,3 @@ "use strict";

return res;
}, new scope_locals_1.ScopeLocals());
}, new scope_locals_1.ScopeLocals({ mode: 'explicit' }));
}

@@ -182,2 +182,2 @@ parseEval(invokedName, path) {

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"expression-parser.js","sourceRoot":"","sources":["expression-parser.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 { ScopeLocals } from './scope-locals';\n\nexport class ExpressionParser {\n  constructor(private babel: typeof Babel) {}\n\n  parseExpression(invokedName: string, path: NodePath<t.Expression>): unknown {\n    switch (path.node.type) {\n      case 'ObjectExpression':\n        return this.parseObjectExpression(invokedName, path as NodePath<t.ObjectExpression>);\n      case 'ArrayExpression': {\n        return this.parseArrayExpression(invokedName, path as NodePath<t.ArrayExpression>);\n      }\n      case 'StringLiteral':\n      case 'BooleanLiteral':\n      case 'NumericLiteral':\n        return path.node.value;\n      default:\n        throw path.buildCodeFrameError(\n          `${invokedName} can only accept static options but you passed ${JSON.stringify(\n            path.node\n          )}`\n        );\n    }\n  }\n\n  parseArrayExpression(invokedName: string, path: NodePath<t.ArrayExpression>) {\n    return path.get('elements').map((element) => {\n      if (element.isSpreadElement()) {\n        throw element.buildCodeFrameError(`spread element is not allowed here`);\n      } else if (element.isExpression()) {\n        return this.parseExpression(invokedName, element);\n      }\n    });\n  }\n\n  parseScope(invokedName: string, path: NodePath<t.ObjectProperty | t.ObjectMethod>): ScopeLocals {\n    let body: t.BlockStatement | t.Expression | undefined = undefined;\n\n    if (path.node.type === 'ObjectMethod') {\n      body = path.node.body;\n    } else {\n      let { value } = path.node;\n      if (this.t.isObjectExpression(value)) {\n        throw path.buildCodeFrameError(\n          `Passing an object as the \\`scope\\` property to inline templates is no longer supported. Please pass a function that returns an object expression instead.`\n        );\n      }\n      if (this.t.isFunctionExpression(value) || this.t.isArrowFunctionExpression(value)) {\n        body = value.body;\n      }\n    }\n\n    let objExpression: t.Expression | undefined | null = undefined;\n\n    if (body?.type === 'ObjectExpression') {\n      objExpression = body;\n    } else if (body?.type === 'BlockStatement') {\n      // SAFETY: We know that the body is a ReturnStatement because we're checking inside\n      let returnStatements = body.body.filter(\n        (statement) => statement.type === 'ReturnStatement'\n      ) as Babel.types.ReturnStatement[];\n\n      if (returnStatements.length !== 1) {\n        throw new Error(\n          'Scope functions must have a single return statement which returns an object expression containing references to in-scope values'\n        );\n      }\n\n      objExpression = returnStatements[0].argument;\n    }\n\n    if (objExpression?.type !== 'ObjectExpression') {\n      throw path.buildCodeFrameError(\n        `Scope objects for \\`${invokedName}\\` must be an object expression containing only references to in-scope values, or a function that returns an object expression containing only references to in-scope values`\n      );\n    }\n\n    return objExpression.properties.reduce((res, prop) => {\n      if (this.t.isSpreadElement(prop)) {\n        throw path.buildCodeFrameError(\n          `Scope objects for \\`${invokedName}\\` may not contain spread elements`\n        );\n      }\n      if (this.t.isObjectMethod(prop)) {\n        throw path.buildCodeFrameError(\n          `Scope objects for \\`${invokedName}\\` may not contain methods`\n        );\n      }\n\n      let { key, value } = prop;\n      if (!this.t.isStringLiteral(key) && !this.t.isIdentifier(key)) {\n        throw path.buildCodeFrameError(\n          `Scope objects for \\`${invokedName}\\` may only contain static property names`\n        );\n      }\n\n      let propName = name(key);\n\n      if (value.type !== 'Identifier') {\n        throw path.buildCodeFrameError(\n          `Scope objects for \\`${invokedName}\\` may only contain direct references to in-scope values, e.g. { ${propName} } or { ${propName}: ${propName} }`\n        );\n      }\n\n      res.add(propName, value.name);\n      return res;\n    }, new ScopeLocals());\n  }\n\n  parseEval(\n    invokedName: string,\n    path: NodePath<t.ObjectProperty | t.ObjectMethod>\n  ): { isEval: true } {\n    let body: NodePath<t.BlockStatement>;\n\n    if (path.isObjectMethod()) {\n      body = path.get('body');\n    } else if (path.isObjectProperty()) {\n      let value = path.get('value');\n      if (value.isFunctionExpression()) {\n        body = value.get('body');\n      } else {\n        throw path.buildCodeFrameError(\n          `unsupported syntax for \\`eval\\` parameter to \\`${invokedName}\\`. It must be an object method or a function.`\n        );\n      }\n    } else {\n      throw path.buildCodeFrameError(\n        `unsupported syntax for \\`eval\\` parameter to \\`${invokedName}\\`. It must be an object method or a function.`\n      );\n    }\n\n    let returnStatements = body\n      .get('body')\n      .filter((statement) => statement.isReturnStatement()) as NodePath<t.ReturnStatement>[];\n\n    if (returnStatements.length !== 1) {\n      throw body.buildCodeFrameError('eval function must have a single return statement');\n    }\n\n    let returnExpression = returnStatements[0].get('argument');\n\n    if (!returnExpression.isCallExpression()) {\n      throw returnStatements[0].buildCodeFrameError(\n        'eval function must return `eval(arguments[0])`. Found non-CallExpression.'\n      );\n    }\n\n    let callee = returnExpression.get('callee');\n    if (!callee.isIdentifier() || callee.node.name !== 'eval') {\n      throw returnExpression.buildCodeFrameError(\n        'eval function must return `eval(arguments[0])`. Found callee is not eval.'\n      );\n    }\n\n    let args = returnExpression.get('arguments');\n    if (args.length !== 1) {\n      throw returnExpression.buildCodeFrameError(\n        'eval function must return `eval(arguments[0])`. Found incorrect number of arguments.'\n      );\n    }\n    let arg = args[0];\n    if (!arg.isMemberExpression()) {\n      throw arg.buildCodeFrameError(\n        'eval function must return `eval(arguments[0])`. Found argument is non-MemberExpression.'\n      );\n    }\n    let obj = arg.get('object');\n    if (!obj.isIdentifier() || obj.node.name !== 'arguments') {\n      throw obj.buildCodeFrameError(\n        'eval function must return `eval(arguments[0])`. Found wrong argument to eval.'\n      );\n    }\n    let prop = arg.get('property');\n    if (!prop.isNumericLiteral() || prop.node.value !== 0) {\n      throw prop.buildCodeFrameError(\n        'eval function must return `eval(arguments[0])`. Found wrong property.'\n      );\n    }\n    return { isEval: true };\n  }\n\n  parseObjectExpression(\n    invokedName: string,\n    path: NodePath<t.ObjectExpression>,\n    shouldParseScope = false,\n    shouldSupportRFC931 = false\n  ) {\n    let result: Record<string, unknown> = {};\n\n    path.get('properties').forEach((property) => {\n      let { node } = property;\n      if (this.t.isSpreadElement(node)) {\n        throw property.buildCodeFrameError(`${invokedName} does not allow spread element`);\n      }\n\n      if (node.computed) {\n        throw property.buildCodeFrameError(`${invokedName} can only accept static property names`);\n      }\n\n      let { key } = node;\n      if (!this.t.isIdentifier(key) && !this.t.isStringLiteral(key)) {\n        throw property.buildCodeFrameError(`${invokedName} can only accept static property names`);\n      }\n\n      let propertyName = name(key);\n\n      if (shouldParseScope && propertyName === 'scope') {\n        result.scope = this.parseScope(invokedName, property as NodePath<typeof node>);\n      } else if (shouldSupportRFC931 && propertyName === 'eval') {\n        result.eval = this.parseEval(invokedName, property as NodePath<typeof node>);\n      } else if (shouldSupportRFC931 && propertyName === 'component') {\n        result.component = (property as NodePath<typeof node>).get('value');\n      } else {\n        if (this.t.isObjectMethod(node)) {\n          throw property.buildCodeFrameError(\n            `${invokedName} does not accept a method for ${propertyName}`\n          );\n        }\n        let valuePath = (property as NodePath<typeof node>).get('value');\n        if (!valuePath.isExpression()) {\n          throw valuePath.buildCodeFrameError(`must be an expression`);\n        }\n        result[propertyName] = this.parseExpression(invokedName, valuePath);\n      }\n    });\n\n    return result;\n  }\n\n  private get t() {\n    return this.babel.types;\n  }\n}\n\nfunction name(node: t.StringLiteral | t.Identifier): string {\n  if (node.type === 'StringLiteral') {\n    return node.value;\n  } else {\n    return node.name;\n  }\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"expression-parser.js","sourceRoot":"","sources":["expression-parser.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 { ScopeLocals } from './scope-locals';\n\nexport class ExpressionParser {\n  constructor(private babel: typeof Babel) {}\n\n  parseExpression(invokedName: string, path: NodePath<t.Expression>): unknown {\n    switch (path.node.type) {\n      case 'ObjectExpression':\n        return this.parseObjectExpression(invokedName, path as NodePath<t.ObjectExpression>);\n      case 'ArrayExpression': {\n        return this.parseArrayExpression(invokedName, path as NodePath<t.ArrayExpression>);\n      }\n      case 'StringLiteral':\n      case 'BooleanLiteral':\n      case 'NumericLiteral':\n        return path.node.value;\n      default:\n        throw path.buildCodeFrameError(\n          `${invokedName} can only accept static options but you passed ${JSON.stringify(\n            path.node\n          )}`\n        );\n    }\n  }\n\n  parseArrayExpression(invokedName: string, path: NodePath<t.ArrayExpression>) {\n    return path.get('elements').map((element) => {\n      if (element.isSpreadElement()) {\n        throw element.buildCodeFrameError(`spread element is not allowed here`);\n      } else if (element.isExpression()) {\n        return this.parseExpression(invokedName, element);\n      }\n    });\n  }\n\n  parseScope(invokedName: string, path: NodePath<t.ObjectProperty | t.ObjectMethod>): ScopeLocals {\n    let body: t.BlockStatement | t.Expression | undefined = undefined;\n\n    if (path.node.type === 'ObjectMethod') {\n      body = path.node.body;\n    } else {\n      let { value } = path.node;\n      if (this.t.isObjectExpression(value)) {\n        throw path.buildCodeFrameError(\n          `Passing an object as the \\`scope\\` property to inline templates is no longer supported. Please pass a function that returns an object expression instead.`\n        );\n      }\n      if (this.t.isFunctionExpression(value) || this.t.isArrowFunctionExpression(value)) {\n        body = value.body;\n      }\n    }\n\n    let objExpression: t.Expression | undefined | null = undefined;\n\n    if (body?.type === 'ObjectExpression') {\n      objExpression = body;\n    } else if (body?.type === 'BlockStatement') {\n      // SAFETY: We know that the body is a ReturnStatement because we're checking inside\n      let returnStatements = body.body.filter(\n        (statement) => statement.type === 'ReturnStatement'\n      ) as Babel.types.ReturnStatement[];\n\n      if (returnStatements.length !== 1) {\n        throw new Error(\n          'Scope functions must have a single return statement which returns an object expression containing references to in-scope values'\n        );\n      }\n\n      objExpression = returnStatements[0].argument;\n    }\n\n    if (objExpression?.type !== 'ObjectExpression') {\n      throw path.buildCodeFrameError(\n        `Scope objects for \\`${invokedName}\\` must be an object expression containing only references to in-scope values, or a function that returns an object expression containing only references to in-scope values`\n      );\n    }\n\n    return objExpression.properties.reduce((res, prop) => {\n      if (this.t.isSpreadElement(prop)) {\n        throw path.buildCodeFrameError(\n          `Scope objects for \\`${invokedName}\\` may not contain spread elements`\n        );\n      }\n      if (this.t.isObjectMethod(prop)) {\n        throw path.buildCodeFrameError(\n          `Scope objects for \\`${invokedName}\\` may not contain methods`\n        );\n      }\n\n      let { key, value } = prop;\n      if (!this.t.isStringLiteral(key) && !this.t.isIdentifier(key)) {\n        throw path.buildCodeFrameError(\n          `Scope objects for \\`${invokedName}\\` may only contain static property names`\n        );\n      }\n\n      let propName = name(key);\n\n      if (value.type !== 'Identifier') {\n        throw path.buildCodeFrameError(\n          `Scope objects for \\`${invokedName}\\` may only contain direct references to in-scope values, e.g. { ${propName} } or { ${propName}: ${propName} }`\n        );\n      }\n\n      res.add(propName, value.name);\n      return res;\n    }, new ScopeLocals({ mode: 'explicit' }));\n  }\n\n  parseEval(\n    invokedName: string,\n    path: NodePath<t.ObjectProperty | t.ObjectMethod>\n  ): { isEval: true } {\n    let body: NodePath<t.BlockStatement>;\n\n    if (path.isObjectMethod()) {\n      body = path.get('body');\n    } else if (path.isObjectProperty()) {\n      let value = path.get('value');\n      if (value.isFunctionExpression()) {\n        body = value.get('body');\n      } else {\n        throw path.buildCodeFrameError(\n          `unsupported syntax for \\`eval\\` parameter to \\`${invokedName}\\`. It must be an object method or a function.`\n        );\n      }\n    } else {\n      throw path.buildCodeFrameError(\n        `unsupported syntax for \\`eval\\` parameter to \\`${invokedName}\\`. It must be an object method or a function.`\n      );\n    }\n\n    let returnStatements = body\n      .get('body')\n      .filter((statement) => statement.isReturnStatement()) as NodePath<t.ReturnStatement>[];\n\n    if (returnStatements.length !== 1) {\n      throw body.buildCodeFrameError('eval function must have a single return statement');\n    }\n\n    let returnExpression = returnStatements[0].get('argument');\n\n    if (!returnExpression.isCallExpression()) {\n      throw returnStatements[0].buildCodeFrameError(\n        'eval function must return `eval(arguments[0])`. Found non-CallExpression.'\n      );\n    }\n\n    let callee = returnExpression.get('callee');\n    if (!callee.isIdentifier() || callee.node.name !== 'eval') {\n      throw returnExpression.buildCodeFrameError(\n        'eval function must return `eval(arguments[0])`. Found callee is not eval.'\n      );\n    }\n\n    let args = returnExpression.get('arguments');\n    if (args.length !== 1) {\n      throw returnExpression.buildCodeFrameError(\n        'eval function must return `eval(arguments[0])`. Found incorrect number of arguments.'\n      );\n    }\n    let arg = args[0];\n    if (!arg.isMemberExpression()) {\n      throw arg.buildCodeFrameError(\n        'eval function must return `eval(arguments[0])`. Found argument is non-MemberExpression.'\n      );\n    }\n    let obj = arg.get('object');\n    if (!obj.isIdentifier() || obj.node.name !== 'arguments') {\n      throw obj.buildCodeFrameError(\n        'eval function must return `eval(arguments[0])`. Found wrong argument to eval.'\n      );\n    }\n    let prop = arg.get('property');\n    if (!prop.isNumericLiteral() || prop.node.value !== 0) {\n      throw prop.buildCodeFrameError(\n        'eval function must return `eval(arguments[0])`. Found wrong property.'\n      );\n    }\n    return { isEval: true };\n  }\n\n  parseObjectExpression(\n    invokedName: string,\n    path: NodePath<t.ObjectExpression>,\n    shouldParseScope = false,\n    shouldSupportRFC931 = false\n  ) {\n    let result: Record<string, unknown> = {};\n\n    path.get('properties').forEach((property) => {\n      let { node } = property;\n      if (this.t.isSpreadElement(node)) {\n        throw property.buildCodeFrameError(`${invokedName} does not allow spread element`);\n      }\n\n      if (node.computed) {\n        throw property.buildCodeFrameError(`${invokedName} can only accept static property names`);\n      }\n\n      let { key } = node;\n      if (!this.t.isIdentifier(key) && !this.t.isStringLiteral(key)) {\n        throw property.buildCodeFrameError(`${invokedName} can only accept static property names`);\n      }\n\n      let propertyName = name(key);\n\n      if (shouldParseScope && propertyName === 'scope') {\n        result.scope = this.parseScope(invokedName, property as NodePath<typeof node>);\n      } else if (shouldSupportRFC931 && propertyName === 'eval') {\n        result.eval = this.parseEval(invokedName, property as NodePath<typeof node>);\n      } else if (shouldSupportRFC931 && propertyName === 'component') {\n        result.component = (property as NodePath<typeof node>).get('value');\n      } else {\n        if (this.t.isObjectMethod(node)) {\n          throw property.buildCodeFrameError(\n            `${invokedName} does not accept a method for ${propertyName}`\n          );\n        }\n        let valuePath = (property as NodePath<typeof node>).get('value');\n        if (!valuePath.isExpression()) {\n          throw valuePath.buildCodeFrameError(`must be an expression`);\n        }\n        result[propertyName] = this.parseExpression(invokedName, valuePath);\n      }\n    });\n\n    return result;\n  }\n\n  private get t() {\n    return this.babel.types;\n  }\n}\n\nfunction name(node: t.StringLiteral | t.Identifier): string {\n  if (node.type === 'StringLiteral') {\n    return node.value;\n  } else {\n    return node.name;\n  }\n}\n"]}

@@ -6,3 +6,2 @@ import type { types as t } from '@babel/core';

import type { ImportUtil } from 'babel-import-util';
import { ScopeLocals } from './scope-locals';
interface State {

@@ -14,3 +13,3 @@ program: NodePath<Babel.types.Program>;

#private;
constructor(babel: typeof Babel, state: State, template: NodePath<t.Expression>, scopeLocals: ScopeLocals, importer: ImportUtil);
constructor(babel: typeof Babel, state: State, template: NodePath<t.Expression>, addedBinding: (name: string) => void, importer: ImportUtil);
/**

@@ -17,0 +16,0 @@ * Create a new binding that you can use in your template, initialized with

@@ -13,13 +13,10 @@ "use strict";

};
var _JSUtils_instances, _JSUtils_babel, _JSUtils_state, _JSUtils_template, _JSUtils_scopeLocals, _JSUtils_importer, _JSUtils_emitStatement, _JSUtils_parseExpression, _ExpressionContext_importer, _ExpressionContext_target;
var _JSUtils_instances, _JSUtils_babel, _JSUtils_state, _JSUtils_template, _JSUtils_addedBinding, _JSUtils_importer, _JSUtils_emitStatement, _JSUtils_parseExpression, _ExpressionContext_importer, _ExpressionContext_target;
Object.defineProperty(exports, "__esModule", { value: true });
exports.JSUtils = void 0;
const hbs_utils_1 = require("./hbs-utils");
// This exists to give AST plugins a controlled interface for influencing the
// surrounding Javascript scope
class JSUtils {
constructor(babel, state, template,
// mapping of handlebars identifiers to javascript identifiers, as appears
// in the `scope` argument to precompileTemplate. This is both read and
// write -- we might put more stuff into it.
scopeLocals, importer) {
constructor(babel, state, template, addedBinding, importer) {
_JSUtils_instances.add(this);

@@ -29,3 +26,3 @@ _JSUtils_babel.set(this, void 0);

_JSUtils_template.set(this, void 0);
_JSUtils_scopeLocals.set(this, void 0);
_JSUtils_addedBinding.set(this, void 0);
_JSUtils_importer.set(this, void 0);

@@ -35,3 +32,3 @@ __classPrivateFieldSet(this, _JSUtils_babel, babel, "f");

__classPrivateFieldSet(this, _JSUtils_template, template, "f");
__classPrivateFieldSet(this, _JSUtils_scopeLocals, scopeLocals, "f");
__classPrivateFieldSet(this, _JSUtils_addedBinding, addedBinding, "f");
__classPrivateFieldSet(this, _JSUtils_importer, importer, "f");

@@ -67,5 +64,3 @@ if (!__classPrivateFieldGet(this, _JSUtils_state, "f").lastInsertedPath) {

var _a;
let name = unusedNameLike((_a = opts === null || opts === void 0 ? void 0 : opts.nameHint) !== null && _a !== void 0 ? _a : 'a', (candidate) => __classPrivateFieldGet(this, _JSUtils_template, "f").scope.hasBinding(candidate) ||
__classPrivateFieldGet(this, _JSUtils_scopeLocals, "f").has(candidate) ||
astNodeHasBinding(target, candidate));
let name = unusedNameLike((_a = opts === null || opts === void 0 ? void 0 : opts.nameHint) !== null && _a !== void 0 ? _a : 'a', (candidate) => __classPrivateFieldGet(this, _JSUtils_template, "f").scope.hasBinding(candidate) || (0, hbs_utils_1.astNodeHasBinding)(target, candidate));
let t = __classPrivateFieldGet(this, _JSUtils_babel, "f").types;

@@ -75,4 +70,4 @@ let declaration = __classPrivateFieldGet(this, _JSUtils_instances, "m", _JSUtils_emitStatement).call(this, t.variableDeclaration('let', [

]));
declaration.scope.registerBinding('module', declaration.get('declarations.0'));
__classPrivateFieldGet(this, _JSUtils_scopeLocals, "f").add(name);
declaration.scope.registerBinding('let', declaration.get('declarations.0'));
__classPrivateFieldGet(this, _JSUtils_addedBinding, "f").call(this, name);
return name;

@@ -97,24 +92,22 @@ }

let importedIdentifier = __classPrivateFieldGet(this, _JSUtils_importer, "f").import(__classPrivateFieldGet(this, _JSUtils_template, "f"), moduleSpecifier, exportedName, opts === null || opts === void 0 ? void 0 : opts.nameHint);
// If we're already referencing the imported name from the outer scope and
// it's not shadowed at our target location in the template, we can reuse
// the existing import.
if (__classPrivateFieldGet(this, _JSUtils_scopeLocals, "f").has(importedIdentifier.name) &&
!astNodeHasBinding(target, importedIdentifier.name)) {
// Simple base case: the JS name that's available is also unused at our spot
// in HBS, so just use it.
if (!(0, hbs_utils_1.astNodeHasBinding)(target, importedIdentifier.name)) {
__classPrivateFieldGet(this, _JSUtils_addedBinding, "f").call(this, importedIdentifier.name);
return importedIdentifier.name;
}
let identifier = unusedNameLike(importedIdentifier.name, (candidate) => __classPrivateFieldGet(this, _JSUtils_scopeLocals, "f").has(candidate) || astNodeHasBinding(target, candidate));
if (identifier !== importedIdentifier.name) {
// The importedIdentifier that we have in Javascript is not usable within
// our HBS because it's shadowed by a block param. So we will introduce a
// second name via a variable declaration.
//
// The reason we don't force the import itself to have this name is that
// we might be re-using an existing import, and we don't want to go
// rewriting all of its callsites that are unrelated to us.
let t = __classPrivateFieldGet(this, _JSUtils_babel, "f").types;
__classPrivateFieldGet(this, _JSUtils_instances, "m", _JSUtils_emitStatement).call(this, t.variableDeclaration('let', [
t.variableDeclarator(t.identifier(identifier), importedIdentifier),
]));
}
__classPrivateFieldGet(this, _JSUtils_scopeLocals, "f").add(identifier);
// The importedIdentifier that we have in Javascript is not usable within
// our HBS because it's shadowed by a block param. So we will introduce a
// second name via a variable declaration.
//
// The reason we don't force the import itself to have this name is that
// we might be re-using an existing import, and we don't want to go
// rewriting all of its callsites that are unrelated to us.
let identifier = unusedNameLike(importedIdentifier.name, (candidate) => __classPrivateFieldGet(this, _JSUtils_template, "f").scope.hasBinding(candidate) || (0, hbs_utils_1.astNodeHasBinding)(target, candidate));
let t = __classPrivateFieldGet(this, _JSUtils_babel, "f").types;
let declaration = __classPrivateFieldGet(this, _JSUtils_instances, "m", _JSUtils_emitStatement).call(this, t.variableDeclaration('let', [
t.variableDeclarator(t.identifier(identifier), importedIdentifier),
]));
declaration.scope.registerBinding('let', declaration.get('declarations.0'));
__classPrivateFieldGet(this, _JSUtils_addedBinding, "f").call(this, identifier);
return identifier;

@@ -145,3 +138,3 @@ }

exports.JSUtils = JSUtils;
_JSUtils_babel = new WeakMap(), _JSUtils_state = new WeakMap(), _JSUtils_template = new WeakMap(), _JSUtils_scopeLocals = new WeakMap(), _JSUtils_importer = new WeakMap(), _JSUtils_instances = new WeakSet(), _JSUtils_emitStatement = function _JSUtils_emitStatement(statement) {
_JSUtils_babel = new WeakMap(), _JSUtils_state = new WeakMap(), _JSUtils_template = new WeakMap(), _JSUtils_addedBinding = new WeakMap(), _JSUtils_importer = new WeakMap(), _JSUtils_instances = new WeakSet(), _JSUtils_emitStatement = function _JSUtils_emitStatement(statement) {
if (__classPrivateFieldGet(this, _JSUtils_state, "f").lastInsertedPath) {

@@ -184,23 +177,2 @@ __classPrivateFieldGet(this, _JSUtils_state, "f").lastInsertedPath = __classPrivateFieldGet(this, _JSUtils_state, "f").lastInsertedPath.insertAfter(statement)[0];

}
function astNodeHasBinding(target, name) {
var _a;
let cursor = target;
while (cursor) {
let parentNode = (_a = cursor.parent) === null || _a === void 0 ? void 0 : _a.node;
if ((parentNode === null || parentNode === void 0 ? void 0 : parentNode.type) === 'ElementNode' &&
parentNode.blockParams.includes(name) &&
// an ElementNode's block params are valid only within its children
parentNode.children.includes(cursor.node)) {
return true;
}
if ((parentNode === null || parentNode === void 0 ? void 0 : parentNode.type) === 'Block' &&
parentNode.blockParams.includes(name) &&
// a Block's blockParams are valid only within its body
parentNode.body.includes(cursor.node)) {
return true;
}
cursor = cursor.parent;
}
return false;
}
function body(node) {

@@ -241,2 +213,2 @@ if (node.type === 'File') {

_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 { ASTPluginBuilder, ASTPluginEnvironment, ASTv1, WalkerPath } from '@glimmer/syntax';\nimport type { ImportUtil } from 'babel-import-util';\nimport { ScopeLocals } from './scope-locals';\n\ninterface State {\n  program: NodePath<Babel.types.Program>;\n  lastInsertedPath: NodePath<Babel.types.Statement> | undefined;\n}\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;\n  #template: NodePath<t.Expression>;\n  #scopeLocals: ScopeLocals;\n  #importer: ImportUtil;\n\n  constructor(\n    babel: typeof Babel,\n    state: State,\n    template: NodePath<t.Expression>,\n    // mapping of handlebars identifiers to javascript identifiers, as appears\n    // in the `scope` argument to precompileTemplate. This is both read and\n    // write -- we might put more stuff into it.\n    scopeLocals: ScopeLocals,\n    importer: ImportUtil\n  ) {\n    this.#babel = babel;\n    this.#state = state;\n    this.#template = template;\n    this.#scopeLocals = scopeLocals;\n    this.#importer = importer;\n\n    if (!this.#state.lastInsertedPath) {\n      let target: NodePath<t.Statement> | undefined;\n      for (let statement of this.#state.program.get('body')) {\n        if (!statement.isImportDeclaration()) {\n          break;\n        }\n        target = statement;\n      }\n      if (target) {\n        this.#state.lastInsertedPath = target;\n      }\n    }\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) ||\n        this.#scopeLocals.has(candidate) ||\n        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.#scopeLocals.add(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    // If we're already referencing the imported name from the outer scope and\n    // it's not shadowed at our target location in the template, we can reuse\n    // the existing import.\n    if (\n      this.#scopeLocals.has(importedIdentifier.name) &&\n      !astNodeHasBinding(target, importedIdentifier.name)\n    ) {\n      return importedIdentifier.name;\n    }\n\n    let identifier = unusedNameLike(\n      importedIdentifier.name,\n      (candidate) => this.#scopeLocals.has(candidate) || 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.#scopeLocals.add(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"]}
//# 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 { astNodeHasBinding } from './hbs-utils';\n\ninterface State {\n  program: NodePath<Babel.types.Program>;\n  lastInsertedPath: NodePath<Babel.types.Statement> | undefined;\n}\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;\n  #template: NodePath<t.Expression>;\n  #addedBinding: (name: string) => void;\n  #importer: ImportUtil;\n\n  constructor(\n    babel: typeof Babel,\n    state: State,\n    template: NodePath<t.Expression>,\n    addedBinding: (name: string) => void,\n    importer: ImportUtil\n  ) {\n    this.#babel = babel;\n    this.#state = state;\n    this.#template = template;\n    this.#addedBinding = addedBinding;\n    this.#importer = importer;\n\n    if (!this.#state.lastInsertedPath) {\n      let target: NodePath<t.Statement> | undefined;\n      for (let statement of this.#state.program.get('body')) {\n        if (!statement.isImportDeclaration()) {\n          break;\n        }\n        target = statement;\n      }\n      if (target) {\n        this.#state.lastInsertedPath = target;\n      }\n    }\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('let', declaration.get('declarations.0') as NodePath);\n    this.#addedBinding(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    // Simple base case: the JS name that's available is also unused at our spot\n    // in HBS, so just use it.\n    if (!astNodeHasBinding(target, importedIdentifier.name)) {\n      this.#addedBinding(importedIdentifier.name);\n      return importedIdentifier.name;\n    }\n\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 identifier = unusedNameLike(\n      importedIdentifier.name,\n      (candidate) =>\n        this.#template.scope.hasBinding(candidate) || astNodeHasBinding(target, candidate)\n    );\n    let t = this.#babel.types;\n    let declaration = this.#emitStatement(\n      t.variableDeclaration('let', [\n        t.variableDeclarator(t.identifier(identifier), importedIdentifier),\n      ])\n    );\n    declaration.scope.registerBinding('let', declaration.get('declarations.0') as NodePath);\n\n    this.#addedBinding(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\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"]}

@@ -0,3 +1,12 @@

import type { NodePath } from '@babel/traverse';
import { ASTPluginEnvironment, NodeVisitor } from '@glimmer/syntax';
type Params = {
mode: 'explicit';
} | {
mode: 'implicit';
jsPath: NodePath;
};
export declare class ScopeLocals {
#private;
constructor(params: Params);
get locals(): string[];

@@ -9,3 +18,8 @@ has(key: string): boolean;

entries(): [string, string][];
add(key: string, value?: string): void;
add(hbsName: string, jsName?: string): void;
crawl(): (_env: ASTPluginEnvironment) => {
name: string;
visitor: NodeVisitor;
};
}
export {};
"use strict";
var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, state, kind, f) {
if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter");
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
};
var _ScopeLocals_mapping, _ScopeLocals_locals;
Object.defineProperty(exports, "__esModule", { value: true });
exports.ScopeLocals = void 0;
/*

@@ -18,9 +10,28 @@ This class exists because:

*/
var __classPrivateFieldSet = (this && this.__classPrivateFieldSet) || function (receiver, state, value, kind, f) {
if (kind === "m") throw new TypeError("Private method is not writable");
if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a setter");
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot write private member to an object whose class did not declare it");
return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;
};
var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, state, kind, f) {
if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter");
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
};
var _ScopeLocals_instances, _ScopeLocals_mapping, _ScopeLocals_locals, _ScopeLocals_params, _ScopeLocals_isInJsScope;
Object.defineProperty(exports, "__esModule", { value: true });
exports.ScopeLocals = void 0;
const hbs_utils_1 = require("./hbs-utils");
const read_only_array_1 = require("./read-only-array");
class ScopeLocals {
constructor() {
constructor(params) {
_ScopeLocals_instances.add(this);
_ScopeLocals_mapping.set(this, {});
_ScopeLocals_locals.set(this, []);
_ScopeLocals_params.set(this, void 0);
__classPrivateFieldSet(this, _ScopeLocals_params, params, "f");
}
get locals() {
return __classPrivateFieldGet(this, _ScopeLocals_locals, "f");
return (0, read_only_array_1.readOnlyArray)(__classPrivateFieldGet(this, _ScopeLocals_locals, "f"), 'The only supported way to manipulate locals is via the jsutils API\nhttps://github.com/emberjs/babel-plugin-ember-template-compilation#jsutils-manipulating-javascript-from-within-ast-transforms');
}

@@ -42,11 +53,73 @@ has(key) {

}
add(key, value) {
__classPrivateFieldGet(this, _ScopeLocals_mapping, "f")[key] = value !== null && value !== void 0 ? value : key;
if (!__classPrivateFieldGet(this, _ScopeLocals_locals, "f").includes(key)) {
__classPrivateFieldGet(this, _ScopeLocals_locals, "f").push(key);
add(hbsName, jsName) {
__classPrivateFieldGet(this, _ScopeLocals_mapping, "f")[hbsName] = jsName !== null && jsName !== void 0 ? jsName : hbsName;
if (!__classPrivateFieldGet(this, _ScopeLocals_locals, "f").includes(hbsName)) {
__classPrivateFieldGet(this, _ScopeLocals_locals, "f").push(hbsName);
}
}
// this AST transform discovers all possible upvars in HBS that refer to valid
// bindings in JS, and then depending on the mode adjusts our actual scope bag
// contents.
crawl() {
return (_env) => {
let seen;
return {
name: 'scope-locals-crawl',
visitor: {
Template: {
enter: () => {
seen = new Set();
},
exit: (_node, _path) => {
if (__classPrivateFieldGet(this, _ScopeLocals_params, "f").mode === 'implicit') {
// all hbs upvars that have matching JS bindings go into the
// scope
for (let name of seen) {
if (__classPrivateFieldGet(this, _ScopeLocals_instances, "m", _ScopeLocals_isInJsScope).call(this, name, __classPrivateFieldGet(this, _ScopeLocals_params, "f").jsPath)) {
this.add(name);
}
}
}
else {
// in explicit form, we might prune back the preexising scope in
// the case where another AST transform has eliminated the use
// of the original binding. But we don't add anything new. The
// only way for new bindings to be introduced into scope is for
// another AST transform to explicitly call the jsutils, which
// calls our `add`.
for (let name of Object.keys(__classPrivateFieldGet(this, _ScopeLocals_mapping, "f"))) {
if (!seen.has(name)) {
__classPrivateFieldGet(this, _ScopeLocals_locals, "f").splice(__classPrivateFieldGet(this, _ScopeLocals_locals, "f").indexOf(name), 1);
delete __classPrivateFieldGet(this, _ScopeLocals_mapping, "f")[name];
}
}
}
},
},
PathExpression: (node, path) => {
if (node.head.type !== 'VarHead') {
return;
}
const name = node.head.name;
if (!(0, hbs_utils_1.astNodeHasBinding)(path, name)) {
seen.add(name);
}
},
ElementNode: (node, path) => {
const name = node.tag.split('.')[0];
if (!(0, hbs_utils_1.astNodeHasBinding)(path, name)) {
seen.add(name);
}
},
},
};
};
}
}
exports.ScopeLocals = ScopeLocals;
_ScopeLocals_mapping = new WeakMap(), _ScopeLocals_locals = new WeakMap();
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2NvcGUtbG9jYWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsic2NvcGUtbG9jYWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7OztBQUFBOzs7Ozs7O0VBT0U7QUFDRixNQUFhLFdBQVc7SUFBeEI7UUFDRSwrQkFBbUMsRUFBRSxFQUFDO1FBQ3RDLDhCQUFvQixFQUFFLEVBQUM7SUFnQ3pCLENBQUM7SUE5QkMsSUFBSSxNQUFNO1FBQ1IsT0FBTyx1QkFBQSxJQUFJLDJCQUFRLENBQUM7SUFDdEIsQ0FBQztJQUVELEdBQUcsQ0FBQyxHQUFXO1FBQ2IsT0FBTyxHQUFHLElBQUksdUJBQUEsSUFBSSw0QkFBUyxDQUFDO0lBQzlCLENBQUM7SUFFRCxHQUFHLENBQUMsR0FBVztRQUNiLE9BQU8sdUJBQUEsSUFBSSw0QkFBUyxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQzVCLENBQUM7SUFFRCxPQUFPO1FBQ0wsT0FBTyx1QkFBQSxJQUFJLDJCQUFRLENBQUMsTUFBTSxLQUFLLENBQUMsQ0FBQztJQUNuQyxDQUFDO0lBRUQsY0FBYztRQUNaLE9BQU8sTUFBTSxDQUFDLE9BQU8sQ0FBQyx1QkFBQSxJQUFJLDRCQUFTLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO0lBQ2pFLENBQUM7SUFFRCxPQUFPO1FBQ0wsT0FBTyxNQUFNLENBQUMsT0FBTyxDQUFDLHVCQUFBLElBQUksNEJBQVMsQ0FBQyxDQUFDO0lBQ3ZDLENBQUM7SUFFRCxHQUFHLENBQUMsR0FBVyxFQUFFLEtBQWM7UUFDN0IsdUJBQUEsSUFBSSw0QkFBUyxDQUFDLEdBQUcsQ0FBQyxHQUFHLEtBQUssYUFBTCxLQUFLLGNBQUwsS0FBSyxHQUFJLEdBQUcsQ0FBQztRQUNsQyxJQUFJLENBQUMsdUJBQUEsSUFBSSwyQkFBUSxDQUFDLFFBQVEsQ0FBQyxHQUFHLENBQUMsRUFBRTtZQUMvQix1QkFBQSxJQUFJLDJCQUFRLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO1NBQ3hCO0lBQ0gsQ0FBQztDQUNGO0FBbENELGtDQWtDQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gIFRoaXMgY2xhc3MgZXhpc3RzIGJlY2F1c2U6XG4gICAtIGJlZm9yZSB0ZW1wbGF0ZSBjb21waWxhdGlvbiBzdGFydHMsIHdlIG5lZWQgdG8gcGFzcyBhIGBsb2NhbHNgIGFycmF5IHRvXG4gICAgIGVtYmVyLXRlbXBsYXRlLWNvbXBpbGVyXG4gICAtIHRoZSBKU1V0aWxzIEFQSSBjYW4gbXV0YXRlIHRoZSBzY29wZSBkdXJpbmcgdGVtcGxhdGUgY29tcGlsYXRpb25cbiAgIC0gdGhvc2Ugc2NvcGUgbXV0YXRpb25zIG5lZWQgdG8gdXBkYXRlIGJvdGggdGhlIG9yaWdpbmFsIGBsb2NhbHNgIGFycmF5IGFuZFxuICAgICBvdXIgb3duIG5hbWUgbWFwcGluZywga2VlcGluZyB0aGVtIGluIHN5bmMuXG4qL1xuZXhwb3J0IGNsYXNzIFNjb3BlTG9jYWxzIHtcbiAgI21hcHBpbmc6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4gPSB7fTtcbiAgI2xvY2Fsczogc3RyaW5nW10gPSBbXTtcblxuICBnZXQgbG9jYWxzKCkge1xuICAgIHJldHVybiB0aGlzLiNsb2NhbHM7XG4gIH1cblxuICBoYXMoa2V5OiBzdHJpbmcpOiBib29sZWFuIHtcbiAgICByZXR1cm4ga2V5IGluIHRoaXMuI21hcHBpbmc7XG4gIH1cblxuICBnZXQoa2V5OiBzdHJpbmcpOiBzdHJpbmcge1xuICAgIHJldHVybiB0aGlzLiNtYXBwaW5nW2tleV07XG4gIH1cblxuICBpc0VtcHR5KCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLiNsb2NhbHMubGVuZ3RoID09PSAwO1xuICB9XG5cbiAgbmVlZHNSZW1hcHBpbmcoKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIE9iamVjdC5lbnRyaWVzKHRoaXMuI21hcHBpbmcpLnNvbWUoKFtrLCB2XSkgPT4gayAhPT0gdik7XG4gIH1cblxuICBlbnRyaWVzKCkge1xuICAgIHJldHVybiBPYmplY3QuZW50cmllcyh0aGlzLiNtYXBwaW5nKTtcbiAgfVxuXG4gIGFkZChrZXk6IHN0cmluZywgdmFsdWU/OiBzdHJpbmcpIHtcbiAgICB0aGlzLiNtYXBwaW5nW2tleV0gPSB2YWx1ZSA/PyBrZXk7XG4gICAgaWYgKCF0aGlzLiNsb2NhbHMuaW5jbHVkZXMoa2V5KSkge1xuICAgICAgdGhpcy4jbG9jYWxzLnB1c2goa2V5KTtcbiAgICB9XG4gIH1cbn1cbiJdfQ==
_ScopeLocals_mapping = new WeakMap(), _ScopeLocals_locals = new WeakMap(), _ScopeLocals_params = new WeakMap(), _ScopeLocals_instances = new WeakSet(), _ScopeLocals_isInJsScope = function _ScopeLocals_isInJsScope(hbsName, jsPath) {
var _a;
let jsName = (_a = __classPrivateFieldGet(this, _ScopeLocals_mapping, "f")[hbsName]) !== null && _a !== void 0 ? _a : hbsName;
return ['this', 'globalThis'].includes(jsName) || jsPath.scope.getBinding(jsName);
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scope-locals.js","sourceRoot":"","sources":["scope-locals.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n  This class exists because:\n   - before template compilation starts, we need to pass a `locals` array to\n     ember-template-compiler\n   - the JSUtils API can mutate the scope during template compilation\n   - those scope mutations need to update both the original `locals` array and\n     our own name mapping, keeping them in sync.\n*/\n\nimport type { NodePath } from '@babel/traverse';\nimport { ASTPluginEnvironment, NodeVisitor } from '@glimmer/syntax';\nimport { astNodeHasBinding } from './hbs-utils';\nimport { readOnlyArray } from './read-only-array';\n\n/*\n    `mode` refers to the implicit and explicit formats defined here:\n\n      https://github.com/emberjs/rfcs/blob/9fd6ceac2559bee1c33acf0d7834e675125a4f16/text/0931-template-compiler-api.md#explicit-form\n      https://github.com/emberjs/rfcs/blob/9fd6ceac2559bee1c33acf0d7834e675125a4f16/text/0931-template-compiler-api.md#implicit-form\n\n    This class needs to know the difference because in implicit format, upvars\n    in hbs are automagically connected with outer Javascript bindings, and in\n    explicit form they are not.\n*/\ntype Params =\n  | {\n      mode: 'explicit';\n    }\n  | {\n      mode: 'implicit';\n      jsPath: NodePath;\n    };\n\nexport class ScopeLocals {\n  constructor(params: Params) {\n    this.#params = params;\n  }\n\n  #mapping: Record<string, string> = {};\n  #locals: string[] = [];\n  #params: Params;\n\n  get locals() {\n    return readOnlyArray(\n      this.#locals,\n      'The only supported way to manipulate locals is via the jsutils API\\nhttps://github.com/emberjs/babel-plugin-ember-template-compilation#jsutils-manipulating-javascript-from-within-ast-transforms'\n    );\n  }\n\n  has(key: string): boolean {\n    return key in this.#mapping;\n  }\n\n  get(key: string): string {\n    return this.#mapping[key];\n  }\n\n  isEmpty(): boolean {\n    return this.#locals.length === 0;\n  }\n\n  needsRemapping(): boolean {\n    return Object.entries(this.#mapping).some(([k, v]) => k !== v);\n  }\n\n  entries() {\n    return Object.entries(this.#mapping);\n  }\n\n  add(hbsName: string, jsName?: string) {\n    this.#mapping[hbsName] = jsName ?? hbsName;\n    if (!this.#locals.includes(hbsName)) {\n      this.#locals.push(hbsName);\n    }\n  }\n\n  #isInJsScope(hbsName: string, jsPath: NodePath) {\n    let jsName = this.#mapping[hbsName] ?? hbsName;\n    return ['this', 'globalThis'].includes(jsName) || jsPath.scope.getBinding(jsName);\n  }\n\n  // this AST transform discovers all possible upvars in HBS that refer to valid\n  // bindings in JS, and then depending on the mode adjusts our actual scope bag\n  // contents.\n  crawl() {\n    return (_env: ASTPluginEnvironment): { name: string; visitor: NodeVisitor } => {\n      let seen: Set<string>;\n      return {\n        name: 'scope-locals-crawl',\n        visitor: {\n          Template: {\n            enter: () => {\n              seen = new Set();\n            },\n            exit: (_node, _path) => {\n              if (this.#params.mode === 'implicit') {\n                // all hbs upvars that have matching JS bindings go into the\n                // scope\n                for (let name of seen) {\n                  if (this.#isInJsScope(name, this.#params.jsPath)) {\n                    this.add(name);\n                  }\n                }\n              } else {\n                // in explicit form, we might prune back the preexising scope in\n                // the case where another AST transform has eliminated the use\n                // of the original binding. But we don't add anything new. The\n                // only way for new bindings to be introduced into scope is for\n                // another AST transform to explicitly call the jsutils, which\n                // calls our `add`.\n                for (let name of Object.keys(this.#mapping)) {\n                  if (!seen.has(name)) {\n                    this.#locals.splice(this.#locals.indexOf(name), 1);\n                    delete this.#mapping[name];\n                  }\n                }\n              }\n            },\n          },\n          PathExpression: (node, path) => {\n            if (node.head.type !== 'VarHead') {\n              return;\n            }\n            const name = node.head.name;\n            if (!astNodeHasBinding(path, name)) {\n              seen.add(name);\n            }\n          },\n          ElementNode: (node, path) => {\n            const name = node.tag.split('.')[0];\n            if (!astNodeHasBinding(path, name)) {\n              seen.add(name);\n            }\n          },\n        },\n      };\n    };\n  }\n}\n"]}

Sorry, the diff of this file is too big to display

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc