@glimmer/compiler
Advanced tools
Comparing version 0.76.0 to 0.77.0
@@ -6,2 +6,8 @@ "use strict"; | ||
}); | ||
Object.defineProperty(exports, "defaultId", { | ||
enumerable: true, | ||
get: function () { | ||
return _compiler.defaultId; | ||
} | ||
}); | ||
Object.defineProperty(exports, "precompile", { | ||
@@ -75,2 +81,2 @@ enumerable: true, | ||
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3BhY2thZ2VzL0BnbGltbWVyL2NvbXBpbGVyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBOztBQUNBOztBQVlBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgcHJlY29tcGlsZSwgcHJlY29tcGlsZUpTT04sIFByZWNvbXBpbGVPcHRpb25zIH0gZnJvbSAnLi9saWIvY29tcGlsZXInO1xuZXhwb3J0IHtcbiAgUHJvZ3JhbVN5bWJvbHMsXG4gIGJ1aWxkU3RhdGVtZW50LFxuICBidWlsZFN0YXRlbWVudHMsXG4gIHMsXG4gIGMsXG4gIHVuaWNvZGUsXG4gIE5FV0xJTkUsXG59IGZyb20gJy4vbGliL2J1aWxkZXIvYnVpbGRlcic7XG5leHBvcnQgeyBCdWlsZGVyU3RhdGVtZW50LCBCdWlsZGVyIH0gZnJvbSAnLi9saWIvYnVpbGRlci9idWlsZGVyLWludGVyZmFjZSc7XG5cbi8vIGV4cG9ydGVkIG9ubHkgZm9yIHRlc3RzXG5leHBvcnQgeyBkZWZhdWx0IGFzIFdpcmVGb3JtYXREZWJ1Z2dlciB9IGZyb20gJy4vbGliL3dpcmUtZm9ybWF0LWRlYnVnJztcbiJdLCJzb3VyY2VSb290IjoiIn0= | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3BhY2thZ2VzL0BnbGltbWVyL2NvbXBpbGVyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBOztBQUNBOztBQVlBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgZGVmYXVsdElkLCBwcmVjb21waWxlLCBwcmVjb21waWxlSlNPTiwgUHJlY29tcGlsZU9wdGlvbnMgfSBmcm9tICcuL2xpYi9jb21waWxlcic7XG5leHBvcnQge1xuICBQcm9ncmFtU3ltYm9scyxcbiAgYnVpbGRTdGF0ZW1lbnQsXG4gIGJ1aWxkU3RhdGVtZW50cyxcbiAgcyxcbiAgYyxcbiAgdW5pY29kZSxcbiAgTkVXTElORSxcbn0gZnJvbSAnLi9saWIvYnVpbGRlci9idWlsZGVyJztcbmV4cG9ydCB7IEJ1aWxkZXJTdGF0ZW1lbnQsIEJ1aWxkZXIgfSBmcm9tICcuL2xpYi9idWlsZGVyL2J1aWxkZXItaW50ZXJmYWNlJztcblxuLy8gZXhwb3J0ZWQgb25seSBmb3IgdGVzdHNcbmV4cG9ydCB7IGRlZmF1bHQgYXMgV2lyZUZvcm1hdERlYnVnZ2VyIH0gZnJvbSAnLi9saWIvd2lyZS1mb3JtYXQtZGVidWcnO1xuIl0sInNvdXJjZVJvb3QiOiIifQ== |
@@ -8,2 +8,3 @@ "use strict"; | ||
exports.precompile = precompile; | ||
exports.defaultId = void 0; | ||
@@ -20,2 +21,30 @@ var _syntax = require("@glimmer/syntax"); | ||
const defaultId = (() => { | ||
let req = typeof module === 'object' && typeof module.require === 'function' ? module.require : require; | ||
if (req) { | ||
try { | ||
const crypto = req('crypto'); | ||
let idFn = src => { | ||
let hash = crypto.createHash('sha1'); | ||
hash.update(src, 'utf8'); // trim to 6 bytes of data (2^48 - 1) | ||
return hash.digest('base64').substring(0, 8); | ||
}; | ||
idFn('test'); | ||
return idFn; | ||
} catch (e) {} | ||
} | ||
return function idFn() { | ||
return null; | ||
}; | ||
})(); | ||
exports.defaultId = defaultId; | ||
const defaultOptions = { | ||
id: defaultId | ||
}; | ||
/* | ||
@@ -35,3 +64,4 @@ * Compile a string into a template javascript string. | ||
*/ | ||
function precompileJSON(string, options = {}) { | ||
function precompileJSON(string, options = defaultOptions) { | ||
var _a, _b; | ||
@@ -76,3 +106,3 @@ | ||
function precompile(source, options = {}) { | ||
function precompile(source, options = defaultOptions) { | ||
var _a, _b; | ||
@@ -82,4 +112,6 @@ | ||
let moduleName = (_a = options.meta) === null || _a === void 0 ? void 0 : _a.moduleName; | ||
let idFn = options.id || defaultId; | ||
let blockJSON = JSON.stringify(block); | ||
let templateJSONObject = { | ||
id: idFn(JSON.stringify(options.meta) + blockJSON), | ||
block: blockJSON, | ||
@@ -109,2 +141,2 @@ moduleName: moduleName !== null && moduleName !== void 0 ? moduleName : '(unknown template module)', | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3BhY2thZ2VzL0BnbGltbWVyL2NvbXBpbGVyL2xpYi9jb21waWxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7OztBQU1BOztBQUNBOztBQUVBOztBQUNBOzs7O0FBRUE7Ozs7Ozs7Ozs7Ozs7O0FBY00sU0FBQSxjQUFBLENBQUEsTUFBQSxFQUVKLE9BQUEsR0FGSSxFQUFBLEVBRTJCOzs7QUFFL0IsTUFBSSxNQUFNLEdBQUcsSUFBQSxjQUFBLENBQUEsTUFBQSxFQUFpQixDQUFBLEVBQUEsR0FBRSxPQUFPLENBQVQsSUFBQSxNQUFBLElBQUEsSUFBYyxFQUFBLEtBQUEsS0FBZCxDQUFBLEdBQWMsS0FBZCxDQUFBLEdBQWMsRUFBQSxDQUE1QyxVQUFhLENBQWI7QUFDQSxNQUFJLENBQUEsR0FBQSxFQUFBLE1BQUEsSUFBZ0IsdUJBQVMsTUFBVCxFQUFwQixPQUFvQixDQUFwQjtBQUNBLE1BQUksS0FBSyxHQUFHLG9CQUFLLE1BQUwsRUFBSyxHQUFMLEVBQWlCLENBQUEsRUFBQSxHQUFFLE9BQU8sQ0FBVCxVQUFBLE1BQUEsSUFBQSxJQUFvQixFQUFBLEtBQUEsS0FBcEIsQ0FBQSxHQUFBLEVBQUEsR0FBakIsS0FBQSxFQUFBLEtBQUEsQ0FBdUQsT0FBRCxJQUFZO0FBQzVFLFdBQU8sbUJBQVAsT0FBTyxDQUFQO0FBREYsR0FBWSxDQUFaOztBQUlBLE1BQUE7QUFBQTtBQUFBLElBQXNCO0FBQ3BCLHlCQUFBLEdBQUEsQ0FBQSxhQUFBLEVBQUEsS0FBQTtBQUNEOztBQUVELE1BQUksS0FBSyxDQUFULElBQUEsRUFBZ0I7QUFDZCxXQUFPLENBQUMsS0FBSyxDQUFOLEtBQUEsRUFBUCxNQUFPLENBQVA7QUFERixHQUFBLE1BRU87QUFDTCxVQUFNLEtBQUssQ0FBWCxNQUFBO0FBQ0Q7RUFHSDtBQUNBOzs7QUFDQSxNQUFNLGlCQUFpQixHQUF2QixzQ0FBQTtBQUVBOzs7Ozs7Ozs7Ozs7Ozs7QUFjTSxTQUFBLFVBQUEsQ0FBQSxNQUFBLEVBQXFDLE9BQUEsR0FBckMsRUFBQSxFQUFvRTs7O0FBQ3hFLE1BQUksQ0FBQSxLQUFBLEVBQUEsVUFBQSxJQUFzQixjQUFjLENBQUEsTUFBQSxFQUF4QyxPQUF3QyxDQUF4QztBQUVBLE1BQUksVUFBVSxHQUFBLENBQUEsRUFBQSxHQUFHLE9BQU8sQ0FBVixJQUFBLE1BQUEsSUFBQSxJQUFlLEVBQUEsS0FBQSxLQUFmLENBQUEsR0FBZSxLQUFmLENBQUEsR0FBZSxFQUFBLENBQTdCLFVBQUE7QUFDQSxNQUFJLFNBQVMsR0FBRyxJQUFJLENBQUosU0FBQSxDQUFoQixLQUFnQixDQUFoQjtBQUNBLE1BQUksa0JBQWtCLEdBQW9DO0FBQ3hELElBQUEsS0FBSyxFQURtRCxTQUFBO0FBRXhELElBQUEsVUFBVSxFQUFFLFVBQVUsS0FBVixJQUFBLElBQUEsVUFBVSxLQUFBLEtBQVYsQ0FBQSxHQUFBLFVBQUEsR0FGNEMsMkJBQUE7QUFHeEQ7QUFDQTtBQUNBLElBQUEsS0FBSyxFQUxtRCxpQkFBQTtBQU14RCxJQUFBLFlBQVksRUFBQSxDQUFBLEVBQUEsR0FBRSxPQUFPLENBQVQsVUFBQSxNQUFBLElBQUEsSUFBb0IsRUFBQSxLQUFBLEtBQXBCLENBQUEsR0FBQSxFQUFBLEdBQXdCO0FBTm9CLEdBQTFEOztBQVNBLE1BQUksQ0FBQyxPQUFPLENBQVosVUFBQSxFQUF5QjtBQUN2QixXQUFPLGtCQUFrQixDQUF6QixLQUFBO0FBZnNFLEdBQUEsQ0FrQnhFOzs7QUFDQSxNQUFJLFdBQVcsR0FBRyxJQUFJLENBQUosU0FBQSxDQUFsQixrQkFBa0IsQ0FBbEI7O0FBRUEsTUFBSSxPQUFPLENBQVAsVUFBQSxJQUFzQixVQUFVLENBQVYsTUFBQSxHQUExQixDQUFBLEVBQWlEO0FBQy9DLFFBQUksT0FBTyxHQUFHLFFBQVEsVUFBVSxDQUFWLElBQUEsQ0FBQSxHQUFBLENBQXRCLEdBQUE7QUFFQSxJQUFBLFdBQVcsR0FBRyxXQUFXLENBQVgsT0FBQSxDQUFvQixJQUFJLGlCQUF4QixHQUFBLEVBQWQsT0FBYyxDQUFkO0FBSEYsR0FBQSxNQUlPO0FBQ0wsSUFBQSxXQUFXLEdBQUcsV0FBVyxDQUFYLE9BQUEsQ0FBb0IsSUFBSSxpQkFBeEIsR0FBQSxFQUFkLE1BQWMsQ0FBZDtBQUNEOztBQUVELFNBQUEsV0FBQTtBQUNEIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgU2VyaWFsaXplZFRlbXBsYXRlQmxvY2ssXG4gIFNlcmlhbGl6ZWRUZW1wbGF0ZVdpdGhMYXp5QmxvY2ssXG4gIFRlbXBsYXRlSmF2YXNjcmlwdCxcbn0gZnJvbSAnQGdsaW1tZXIvaW50ZXJmYWNlcyc7XG5pbXBvcnQgeyBMT0NBTF9TSE9VTERfTE9HIH0gZnJvbSAnQGdsaW1tZXIvbG9jYWwtZGVidWctZmxhZ3MnO1xuaW1wb3J0IHsgbm9ybWFsaXplLCBQcmVjb21waWxlT3B0aW9ucywgU291cmNlIH0gZnJvbSAnQGdsaW1tZXIvc3ludGF4JztcbmltcG9ydCB7IExPQ0FMX0xPR0dFUiB9IGZyb20gJ0BnbGltbWVyL3V0aWwnO1xuXG5pbXBvcnQgcGFzczAgZnJvbSAnLi9wYXNzZXMvMS1ub3JtYWxpemF0aW9uL2luZGV4JztcbmltcG9ydCB7IHZpc2l0IGFzIHBhc3MyIH0gZnJvbSAnLi9wYXNzZXMvMi1lbmNvZGluZy9pbmRleCc7XG5cbi8qXG4gKiBDb21waWxlIGEgc3RyaW5nIGludG8gYSB0ZW1wbGF0ZSBqYXZhc2NyaXB0IHN0cmluZy5cbiAqXG4gKiBFeGFtcGxlIHVzYWdlOlxuICogICAgIGltcG9ydCB7IHByZWNvbXBpbGUgfSBmcm9tICdAZ2xpbW1lci9jb21waWxlcic7XG4gKiAgICAgaW1wb3J0IHsgdGVtcGxhdGVGYWN0b3J5IH0gZnJvbSAnZ2xpbW1lci1ydW50aW1lJztcbiAqICAgICBsZXQgdGVtcGxhdGVKcyA9IHByZWNvbXBpbGUoXCJIb3dkeSB7e25hbWV9fVwiKTtcbiAqICAgICBsZXQgZmFjdG9yeSA9IHRlbXBsYXRlRmFjdG9yeShuZXcgRnVuY3Rpb24oXCJyZXR1cm4gXCIgKyB0ZW1wbGF0ZUpzKSgpKTtcbiAqICAgICBsZXQgdGVtcGxhdGUgPSBmYWN0b3J5LmNyZWF0ZShlbnYpO1xuICpcbiAqIEBtZXRob2QgcHJlY29tcGlsZVxuICogQHBhcmFtIHtzdHJpbmd9IHN0cmluZyBhIEdsaW1tZXIgdGVtcGxhdGUgc3RyaW5nXG4gKiBAcmV0dXJuIHtzdHJpbmd9IGEgdGVtcGxhdGUgamF2YXNjcmlwdCBzdHJpbmdcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHByZWNvbXBpbGVKU09OKFxuICBzdHJpbmc6IHN0cmluZyxcbiAgb3B0aW9uczogUHJlY29tcGlsZU9wdGlvbnMgPSB7fVxuKTogW2Jsb2NrOiBTZXJpYWxpemVkVGVtcGxhdGVCbG9jaywgdXNlZExvY2Fsczogc3RyaW5nW11dIHtcbiAgbGV0IHNvdXJjZSA9IG5ldyBTb3VyY2Uoc3RyaW5nLCBvcHRpb25zLm1ldGE/Lm1vZHVsZU5hbWUpO1xuICBsZXQgW2FzdCwgbG9jYWxzXSA9IG5vcm1hbGl6ZShzb3VyY2UsIG9wdGlvbnMpO1xuICBsZXQgYmxvY2sgPSBwYXNzMChzb3VyY2UsIGFzdCwgb3B0aW9ucy5zdHJpY3RNb2RlID8/IGZhbHNlKS5tYXBPaygocGFzczJJbikgPT4ge1xuICAgIHJldHVybiBwYXNzMihwYXNzMkluKTtcbiAgfSk7XG5cbiAgaWYgKExPQ0FMX1NIT1VMRF9MT0cpIHtcbiAgICBMT0NBTF9MT0dHRVIubG9nKGBUZW1wbGF0ZSAtPmAsIGJsb2NrKTtcbiAgfVxuXG4gIGlmIChibG9jay5pc09rKSB7XG4gICAgcmV0dXJuIFtibG9jay52YWx1ZSwgbG9jYWxzXTtcbiAgfSBlbHNlIHtcbiAgICB0aHJvdyBibG9jay5yZWFzb247XG4gIH1cbn1cblxuLy8gVVVJRCB1c2VkIGFzIGEgdW5pcXVlIHBsYWNlaG9sZGVyIGZvciBwbGFjaW5nIGEgc25pcHBldCBvZiBKUyBjb2RlIGludG9cbi8vIHRoZSBvdGhlcndpc2UgSlNPTiBzdHJpbmdpZmllZCB2YWx1ZSBiZWxvdy5cbmNvbnN0IFNDT1BFX1BMQUNFSE9MREVSID0gJzc5NmQyNGU2LTI0NTAtNGZiMC04Y2RmLWI2NTYzOGI1ZWY3MCc7XG5cbi8qXG4gKiBDb21waWxlIGEgc3RyaW5nIGludG8gYSB0ZW1wbGF0ZSBqYXZhc2NyaXB0IHN0cmluZy5cbiAqXG4gKiBFeGFtcGxlIHVzYWdlOlxuICogICAgIGltcG9ydCB7IHByZWNvbXBpbGUgfSBmcm9tICdAZ2xpbW1lci9jb21waWxlcic7XG4gKiAgICAgaW1wb3J0IHsgdGVtcGxhdGVGYWN0b3J5IH0gZnJvbSAnZ2xpbW1lci1ydW50aW1lJztcbiAqICAgICBsZXQgdGVtcGxhdGVKcyA9IHByZWNvbXBpbGUoXCJIb3dkeSB7e25hbWV9fVwiKTtcbiAqICAgICBsZXQgZmFjdG9yeSA9IHRlbXBsYXRlRmFjdG9yeShuZXcgRnVuY3Rpb24oXCJyZXR1cm4gXCIgKyB0ZW1wbGF0ZUpzKSgpKTtcbiAqICAgICBsZXQgdGVtcGxhdGUgPSBmYWN0b3J5LmNyZWF0ZShlbnYpO1xuICpcbiAqIEBtZXRob2QgcHJlY29tcGlsZVxuICogQHBhcmFtIHtzdHJpbmd9IHN0cmluZyBhIEdsaW1tZXIgdGVtcGxhdGUgc3RyaW5nXG4gKiBAcmV0dXJuIHtzdHJpbmd9IGEgdGVtcGxhdGUgamF2YXNjcmlwdCBzdHJpbmdcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHByZWNvbXBpbGUoc291cmNlOiBzdHJpbmcsIG9wdGlvbnM6IFByZWNvbXBpbGVPcHRpb25zID0ge30pOiBUZW1wbGF0ZUphdmFzY3JpcHQge1xuICBsZXQgW2Jsb2NrLCB1c2VkTG9jYWxzXSA9IHByZWNvbXBpbGVKU09OKHNvdXJjZSwgb3B0aW9ucyk7XG5cbiAgbGV0IG1vZHVsZU5hbWUgPSBvcHRpb25zLm1ldGE/Lm1vZHVsZU5hbWU7XG4gIGxldCBibG9ja0pTT04gPSBKU09OLnN0cmluZ2lmeShibG9jayk7XG4gIGxldCB0ZW1wbGF0ZUpTT05PYmplY3Q6IFNlcmlhbGl6ZWRUZW1wbGF0ZVdpdGhMYXp5QmxvY2sgPSB7XG4gICAgYmxvY2s6IGJsb2NrSlNPTixcbiAgICBtb2R1bGVOYW1lOiBtb2R1bGVOYW1lID8/ICcodW5rbm93biB0ZW1wbGF0ZSBtb2R1bGUpJyxcbiAgICAvLyBseWluZyB0byB0aGUgdHlwZSBjaGVja2VyIGhlcmUgYmVjYXVzZSB3ZSdyZSBnb2luZyB0b1xuICAgIC8vIHJlcGxhY2UgaXQganVzdCBiZWxvdywgYWZ0ZXIgc3RyaW5naWZpY2F0aW9uXG4gICAgc2NvcGU6IChTQ09QRV9QTEFDRUhPTERFUiBhcyB1bmtub3duKSBhcyBudWxsLFxuICAgIGlzU3RyaWN0TW9kZTogb3B0aW9ucy5zdHJpY3RNb2RlID8/IGZhbHNlLFxuICB9O1xuXG4gIGlmICghb3B0aW9ucy5zdHJpY3RNb2RlKSB7XG4gICAgZGVsZXRlIHRlbXBsYXRlSlNPTk9iamVjdC5zY29wZTtcbiAgfVxuXG4gIC8vIEpTT04gaXMgamF2YXNjcmlwdFxuICBsZXQgc3RyaW5naWZpZWQgPSBKU09OLnN0cmluZ2lmeSh0ZW1wbGF0ZUpTT05PYmplY3QpO1xuXG4gIGlmIChvcHRpb25zLnN0cmljdE1vZGUgJiYgdXNlZExvY2Fscy5sZW5ndGggPiAwKSB7XG4gICAgbGV0IHNjb3BlRm4gPSBgKCk9Plske3VzZWRMb2NhbHMuam9pbignLCcpfV1gO1xuXG4gICAgc3RyaW5naWZpZWQgPSBzdHJpbmdpZmllZC5yZXBsYWNlKGBcIiR7U0NPUEVfUExBQ0VIT0xERVJ9XCJgLCBzY29wZUZuKTtcbiAgfSBlbHNlIHtcbiAgICBzdHJpbmdpZmllZCA9IHN0cmluZ2lmaWVkLnJlcGxhY2UoYFwiJHtTQ09QRV9QTEFDRUhPTERFUn1cImAsICdudWxsJyk7XG4gIH1cblxuICByZXR1cm4gc3RyaW5naWZpZWQ7XG59XG5cbmV4cG9ydCB7IFByZWNvbXBpbGVPcHRpb25zIH07XG4iXSwic291cmNlUm9vdCI6IiJ9 | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../packages/@glimmer/compiler/lib/compiler.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  SerializedTemplateBlock,\n  SerializedTemplateWithLazyBlock,\n  TemplateJavascript,\n} from '@glimmer/interfaces';\nimport { LOCAL_SHOULD_LOG } from '@glimmer/local-debug-flags';\nimport { normalize, PrecompileOptions, Source, TemplateIdFn } from '@glimmer/syntax';\nimport { LOCAL_LOGGER } from '@glimmer/util';\n\nimport pass0 from './passes/1-normalization/index';\nimport { visit as pass2 } from './passes/2-encoding/index';\n\ndeclare function require(id: 'crypto'): Crypto;\ndeclare function require(id: string): unknown;\n\ninterface Crypto {\n  createHash(\n    alg: 'sha1'\n  ): {\n    update(src: string, encoding: 'utf8'): void;\n    digest(encoding: 'base64'): string;\n  };\n}\n\nexport const defaultId: TemplateIdFn = (() => {\n  let req: typeof require | undefined =\n    typeof module === 'object' && typeof module.require === 'function' ? module.require : require;\n\n  if (req) {\n    try {\n      const crypto = req('crypto');\n\n      let idFn: TemplateIdFn = (src) => {\n        let hash = crypto.createHash('sha1');\n        hash.update(src, 'utf8');\n        // trim to 6 bytes of data (2^48 - 1)\n        return hash.digest('base64').substring(0, 8);\n      };\n\n      idFn('test');\n\n      return idFn;\n    } catch (e) {}\n  }\n\n  return function idFn() {\n    return null;\n  };\n})();\n\nconst defaultOptions: PrecompileOptions = {\n  id: defaultId,\n};\n\n/*\n * Compile a string into a template javascript string.\n *\n * Example usage:\n *     import { precompile } from '@glimmer/compiler';\n *     import { templateFactory } from 'glimmer-runtime';\n *     let templateJs = precompile(\"Howdy {{name}}\");\n *     let factory = templateFactory(new Function(\"return \" + templateJs)());\n *     let template = factory.create(env);\n *\n * @method precompile\n * @param {string} string a Glimmer template string\n * @return {string} a template javascript string\n */\nexport function precompileJSON(\n  string: string,\n  options: PrecompileOptions = defaultOptions\n): [block: SerializedTemplateBlock, usedLocals: string[]] {\n  let source = new Source(string, options.meta?.moduleName);\n  let [ast, locals] = normalize(source, options);\n  let block = pass0(source, ast, options.strictMode ?? false).mapOk((pass2In) => {\n    return pass2(pass2In);\n  });\n\n  if (LOCAL_SHOULD_LOG) {\n    LOCAL_LOGGER.log(`Template ->`, block);\n  }\n\n  if (block.isOk) {\n    return [block.value, locals];\n  } else {\n    throw block.reason;\n  }\n}\n\n// UUID used as a unique placeholder for placing a snippet of JS code into\n// the otherwise JSON stringified value below.\nconst SCOPE_PLACEHOLDER = '796d24e6-2450-4fb0-8cdf-b65638b5ef70';\n\n/*\n * Compile a string into a template javascript string.\n *\n * Example usage:\n *     import { precompile } from '@glimmer/compiler';\n *     import { templateFactory } from 'glimmer-runtime';\n *     let templateJs = precompile(\"Howdy {{name}}\");\n *     let factory = templateFactory(new Function(\"return \" + templateJs)());\n *     let template = factory.create(env);\n *\n * @method precompile\n * @param {string} string a Glimmer template string\n * @return {string} a template javascript string\n */\nexport function precompile(\n  source: string,\n  options: PrecompileOptions = defaultOptions\n): TemplateJavascript {\n  let [block, usedLocals] = precompileJSON(source, options);\n\n  let moduleName = options.meta?.moduleName;\n  let idFn = options.id || defaultId;\n  let blockJSON = JSON.stringify(block);\n  let templateJSONObject: SerializedTemplateWithLazyBlock = {\n    id: idFn(JSON.stringify(options.meta) + blockJSON),\n    block: blockJSON,\n    moduleName: moduleName ?? '(unknown template module)',\n    // lying to the type checker here because we're going to\n    // replace it just below, after stringification\n    scope: (SCOPE_PLACEHOLDER as unknown) as null,\n    isStrictMode: options.strictMode ?? false,\n  };\n\n  if (!options.strictMode) {\n    delete templateJSONObject.scope;\n  }\n\n  // JSON is javascript\n  let stringified = JSON.stringify(templateJSONObject);\n\n  if (options.strictMode && usedLocals.length > 0) {\n    let scopeFn = `()=>[${usedLocals.join(',')}]`;\n\n    stringified = stringified.replace(`\"${SCOPE_PLACEHOLDER}\"`, scopeFn);\n  } else {\n    stringified = stringified.replace(`\"${SCOPE_PLACEHOLDER}\"`, 'null');\n  }\n\n  return stringified;\n}\n\nexport { PrecompileOptions };\n"],"sourceRoot":""} |
@@ -53,5 +53,5 @@ "use strict"; | ||
let path = getPathExpression(node); | ||
let path = getCalleeExpression(node); | ||
if (path !== null && path.ref.type === 'Free') { | ||
if (path !== null && path.type === 'Path' && path.ref.type === 'Free') { | ||
if (path.tail.length > 0) { | ||
@@ -73,5 +73,5 @@ if (path.ref.resolution.serialize() === 'Loose') { | ||
if (this.match(node)) { | ||
let path = getPathExpression(node); | ||
let path = getCalleeExpression(node); | ||
if (path !== null && path.tail.length > 0) { | ||
if (path !== null && path.type === 'Path' && path.tail.length > 0) { | ||
return (0, _result.Err)((0, _syntax.generateSyntaxError)(`The \`${this.keyword}\` keyword was used incorrectly. It was used as \`${path.loc.asString()}\`, but it cannot be used with additional path segments. \n\nError caused by`, node.loc)); | ||
@@ -104,3 +104,3 @@ } | ||
function getPathExpression(node) { | ||
function getCalleeExpression(node) { | ||
switch (node.type) { | ||
@@ -113,3 +113,3 @@ // This covers the inside of attributes and expressions, as well as the callee | ||
case 'AppendContent': | ||
return getPathExpression(node.value); | ||
return getCalleeExpression(node.value); | ||
@@ -119,3 +119,3 @@ case 'Call': | ||
case 'ElementModifier': | ||
return getPathExpression(node.callee); | ||
return node.callee; | ||
@@ -151,5 +151,5 @@ default: | ||
let path = getPathExpression(node); | ||
let path = getCalleeExpression(node); | ||
if (path && path.ref.type === 'Free' && (0, _syntax.isKeyword)(path.ref.name)) { | ||
if (path && path.type === 'Path' && path.ref.type === 'Free' && (0, _syntax.isKeyword)(path.ref.name)) { | ||
let { | ||
@@ -288,2 +288,2 @@ name | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../../../packages/@glimmer/compiler/lib/passes/1-normalization/keywords/impl.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ASTv2,\n  generateSyntaxError,\n  isKeyword,\n  KEYWORDS_TYPES,\n  KeywordType,\n} from '@glimmer/syntax';\nimport { exhausted } from '@glimmer/util';\n\nimport { Err, Result } from '../../../shared/result';\nimport { NormalizationState } from '../context';\n\nexport interface KeywordDelegate<Match extends KeywordMatch, V, Out> {\n  assert(options: Match, state: NormalizationState): Result<V>;\n  translate(options: { node: Match; state: NormalizationState }, param: V): Result<Out>;\n}\n\nexport interface Keyword<K extends KeywordType = KeywordType, Out = unknown> {\n  translate(node: KeywordCandidates[K], state: NormalizationState): Result<Out> | null;\n}\n\nexport interface BlockKeyword<Out = unknown> {\n  translate(node: ASTv2.InvokeBlock, state: NormalizationState): Result<Out> | null;\n}\n\nclass KeywordImpl<\n  K extends KeywordType,\n  S extends string = string,\n  Param = unknown,\n  Out = unknown\n> {\n  protected types: Set<KeywordCandidates[K]['type']>;\n\n  constructor(\n    protected keyword: S,\n    type: KeywordType,\n    private delegate: KeywordDelegate<KeywordMatches[K], Param, Out>\n  ) {\n    let nodes = new Set<KeywordNode['type']>();\n    for (let nodeType of KEYWORD_NODES[type]) {\n      nodes.add(nodeType);\n    }\n\n    this.types = nodes;\n  }\n\n  protected match(node: KeywordCandidates[K]): node is KeywordMatches[K] {\n    if (!this.types.has(node.type)) {\n      return false;\n    }\n\n    let path = getPathExpression(node);\n\n    if (path !== null && path.ref.type === 'Free') {\n      if (path.tail.length > 0) {\n        if (path.ref.resolution.serialize() === 'Loose') {\n          // cannot be a keyword reference, keywords do not allow paths (must be\n          // relying on implicit this fallback)\n          return false;\n        }\n      }\n\n      return path.ref.name === this.keyword;\n    } else {\n      return false;\n    }\n  }\n\n  translate(node: KeywordMatches[K], state: NormalizationState): Result<Out> | null {\n    if (this.match(node)) {\n      let path = getPathExpression(node);\n\n      if (path !== null && path.tail.length > 0) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${\n              this.keyword\n            }\\` keyword was used incorrectly. It was used as \\`${path.loc.asString()}\\`, but it cannot be used with additional path segments. \\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n\n      let param = this.delegate.assert(node, state);\n      return param.andThen((param) => this.delegate.translate({ node, state }, param));\n    } else {\n      return null;\n    }\n  }\n}\n\nexport type PossibleNode =\n  | ASTv2.PathExpression\n  | ASTv2.AppendContent\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock;\n\nexport const KEYWORD_NODES = {\n  Call: ['Call'],\n  Block: ['InvokeBlock'],\n  Append: ['AppendContent'],\n  Modifier: ['ElementModifier'],\n} as const;\n\nexport interface KeywordCandidates {\n  Call: ASTv2.ExpressionNode;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordCandidate = KeywordCandidates[keyof KeywordCandidates];\n\nexport interface KeywordMatches {\n  Call: ASTv2.CallExpression;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordMatch = KeywordMatches[keyof KeywordMatches];\n\n/**\n * A \"generic\" keyword is something like `has-block`, which makes sense in the context\n * of sub-expression or append\n */\nexport type GenericKeywordNode = ASTv2.AppendContent | ASTv2.CallExpression;\n\nexport type KeywordNode =\n  | GenericKeywordNode\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock\n  | ASTv2.ElementModifier;\n\nexport function keyword<\n  K extends KeywordType,\n  D extends KeywordDelegate<KeywordMatches[K], unknown, Out>,\n  Out = unknown\n>(keyword: string, type: K, delegate: D): Keyword<K, Out> {\n  return new KeywordImpl(keyword, type, delegate as KeywordDelegate<KeywordMatch, unknown, Out>);\n}\n\nexport type PossibleKeyword = KeywordNode;\ntype OutFor<K extends Keyword | BlockKeyword> = K extends BlockKeyword<infer Out>\n  ? Out\n  : K extends Keyword<KeywordType, infer Out>\n  ? Out\n  : never;\n\nfunction getPathExpression(node: KeywordNode | ASTv2.ExpressionNode): ASTv2.PathExpression | null {\n  switch (node.type) {\n    // This covers the inside of attributes and expressions, as well as the callee\n    // of call nodes\n    case 'Path':\n      return node;\n    case 'AppendContent':\n      return getPathExpression(node.value);\n    case 'Call':\n    case 'InvokeBlock':\n    case 'ElementModifier':\n      return getPathExpression(node.callee);\n    default:\n      return null;\n  }\n}\n\nexport class Keywords<K extends KeywordType, KeywordList extends Keyword<K> = never>\n  implements Keyword<K, OutFor<KeywordList>> {\n  #keywords: Keyword[] = [];\n  #type: K;\n\n  constructor(type: K) {\n    this.#type = type;\n  }\n\n  kw<S extends string = string, Out = unknown>(\n    name: S,\n    delegate: KeywordDelegate<KeywordMatches[K], unknown, Out>\n  ): Keywords<K, KeywordList | Keyword<K, Out>> {\n    this.#keywords.push(keyword(name, this.#type, delegate));\n\n    return this;\n  }\n\n  translate(\n    node: KeywordCandidates[K],\n    state: NormalizationState\n  ): Result<OutFor<KeywordList>> | null {\n    for (let keyword of this.#keywords) {\n      let result = keyword.translate(node, state) as Result<OutFor<KeywordList>>;\n      if (result !== null) {\n        return result;\n      }\n    }\n\n    let path = getPathExpression(node);\n\n    if (path && path.ref.type === 'Free' && isKeyword(path.ref.name)) {\n      let { name } = path.ref;\n\n      let usedType = this.#type;\n      let validTypes = KEYWORDS_TYPES[name];\n\n      if (validTypes.indexOf(usedType) === -1) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${name}\\` keyword was used incorrectly. It was used as ${\n              typesToReadableName[usedType]\n            }, but its valid usages are:\\n\\n${generateTypesMessage(\n              name,\n              validTypes\n            )}\\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n    }\n\n    return null;\n  }\n}\n\nconst typesToReadableName = {\n  Append: 'an append statement',\n  Block: 'a block statement',\n  Call: 'a call expression',\n  Modifier: 'a modifier',\n};\n\nfunction generateTypesMessage(name: string, types: KeywordType[]): string {\n  return types\n    .map((type) => {\n      switch (type) {\n        case 'Append':\n          return `- As an append statement, as in: {{${name}}}`;\n        case 'Block':\n          return `- As a block statement, as in: {{#${name}}}{{/${name}}}`;\n        case 'Call':\n          return `- As an expression, as in: (${name})`;\n        case 'Modifier':\n          return `- As a modifier, as in: <div {{${name}}}></div>`;\n        default:\n          return exhausted(type);\n      }\n    })\n    .join('\\n\\n');\n}\n\n/**\n * This function builds keyword definitions for a particular type of AST node (`KeywordType`).\n *\n * You can build keyword definitions for:\n *\n * - `Expr`: A `SubExpression` or `PathExpression`\n * - `Block`: A `BlockStatement`\n *   - A `BlockStatement` is a keyword candidate if its head is a\n *     `PathExpression`\n * - `Append`: An `AppendStatement`\n *\n * A node is a keyword candidate if:\n *\n * - A `PathExpression` is a keyword candidate if it has no tail, and its\n *   head expression is a `LocalVarHead` or `FreeVarHead` whose name is\n *   the keyword's name.\n * - A `SubExpression`, `AppendStatement`, or `BlockStatement` is a keyword\n *   candidate if its head is a keyword candidate.\n *\n * The keyword infrastructure guarantees that:\n *\n * - If a node is not a keyword candidate, it is never passed to any keyword's\n *   `assert` method.\n * - If a node is not the `KeywordType` for a particular keyword, it will not\n *   be passed to the keyword's `assert` method.\n *\n * `Expr` keywords are used in expression positions and should return HIR\n * expressions. `Block` and `Append` keywords are used in statement\n * positions and should return HIR statements.\n *\n * A keyword definition has two parts:\n *\n * - `match`, which determines whether an AST node matches the keyword, and can\n *   optionally return some information extracted from the AST node.\n * - `translate`, which takes a matching AST node as well as the extracted\n *   information and returns an appropriate HIR instruction.\n *\n * # Example\n *\n * This keyword:\n *\n * - turns `(hello)` into `\"hello\"`\n *   - as long as `hello` is not in scope\n * - makes it an error to pass any arguments (such as `(hello world)`)\n *\n * ```ts\n * keywords('SubExpr').kw('hello', {\n *   assert(node: ExprKeywordNode): Result<void> | false {\n *     // we don't want to transform `hello` as a `PathExpression`\n *     if (node.type !== 'SubExpression') {\n *       return false;\n *     }\n *\n *     // node.head would be `LocalVarHead` if `hello` was in scope\n *     if (node.head.type !== 'FreeVarHead') {\n *       return false;\n *     }\n *\n *     if (node.params.length || node.hash) {\n *       return Err(generateSyntaxError(`(hello) does not take any arguments`), node.loc);\n *     } else {\n *       return Ok();\n *     }\n *   },\n *\n *   translate(node: ASTv2.SubExpression): hir.Expression {\n *     return ASTv2.builders.literal(\"hello\", node.loc)\n *   }\n * })\n * ```\n *\n * The keyword infrastructure checks to make sure that the node is the right\n * type before calling `assert`, so you only need to consider `SubExpression`\n * and `PathExpression` here. It also checks to make sure that the node passed\n * to `assert` has the keyword name in the right place.\n *\n * Note the important difference between returning `false` from `assert`,\n * which just means that the node didn't match, and returning `Err`, which\n * means that the node matched, but there was a keyword-specific syntax\n * error.\n */\nexport function keywords<K extends KeywordType>(type: K): Keywords<K> {\n  return new Keywords(type);\n}\n"],"sourceRoot":""} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../../../packages/@glimmer/compiler/lib/passes/1-normalization/keywords/impl.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ASTv2,\n  generateSyntaxError,\n  isKeyword,\n  KEYWORDS_TYPES,\n  KeywordType,\n} from '@glimmer/syntax';\nimport { exhausted } from '@glimmer/util';\n\nimport { Err, Result } from '../../../shared/result';\nimport { NormalizationState } from '../context';\n\nexport interface KeywordDelegate<Match extends KeywordMatch, V, Out> {\n  assert(options: Match, state: NormalizationState): Result<V>;\n  translate(options: { node: Match; state: NormalizationState }, param: V): Result<Out>;\n}\n\nexport interface Keyword<K extends KeywordType = KeywordType, Out = unknown> {\n  translate(node: KeywordCandidates[K], state: NormalizationState): Result<Out> | null;\n}\n\nexport interface BlockKeyword<Out = unknown> {\n  translate(node: ASTv2.InvokeBlock, state: NormalizationState): Result<Out> | null;\n}\n\nclass KeywordImpl<\n  K extends KeywordType,\n  S extends string = string,\n  Param = unknown,\n  Out = unknown\n> {\n  protected types: Set<KeywordCandidates[K]['type']>;\n\n  constructor(\n    protected keyword: S,\n    type: KeywordType,\n    private delegate: KeywordDelegate<KeywordMatches[K], Param, Out>\n  ) {\n    let nodes = new Set<KeywordNode['type']>();\n    for (let nodeType of KEYWORD_NODES[type]) {\n      nodes.add(nodeType);\n    }\n\n    this.types = nodes;\n  }\n\n  protected match(node: KeywordCandidates[K]): node is KeywordMatches[K] {\n    if (!this.types.has(node.type)) {\n      return false;\n    }\n\n    let path = getCalleeExpression(node);\n\n    if (path !== null && path.type === 'Path' && path.ref.type === 'Free') {\n      if (path.tail.length > 0) {\n        if (path.ref.resolution.serialize() === 'Loose') {\n          // cannot be a keyword reference, keywords do not allow paths (must be\n          // relying on implicit this fallback)\n          return false;\n        }\n      }\n\n      return path.ref.name === this.keyword;\n    } else {\n      return false;\n    }\n  }\n\n  translate(node: KeywordMatches[K], state: NormalizationState): Result<Out> | null {\n    if (this.match(node)) {\n      let path = getCalleeExpression(node);\n\n      if (path !== null && path.type === 'Path' && path.tail.length > 0) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${\n              this.keyword\n            }\\` keyword was used incorrectly. It was used as \\`${path.loc.asString()}\\`, but it cannot be used with additional path segments. \\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n\n      let param = this.delegate.assert(node, state);\n      return param.andThen((param) => this.delegate.translate({ node, state }, param));\n    } else {\n      return null;\n    }\n  }\n}\n\nexport type PossibleNode =\n  | ASTv2.PathExpression\n  | ASTv2.AppendContent\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock;\n\nexport const KEYWORD_NODES = {\n  Call: ['Call'],\n  Block: ['InvokeBlock'],\n  Append: ['AppendContent'],\n  Modifier: ['ElementModifier'],\n} as const;\n\nexport interface KeywordCandidates {\n  Call: ASTv2.ExpressionNode;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordCandidate = KeywordCandidates[keyof KeywordCandidates];\n\nexport interface KeywordMatches {\n  Call: ASTv2.CallExpression;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordMatch = KeywordMatches[keyof KeywordMatches];\n\n/**\n * A \"generic\" keyword is something like `has-block`, which makes sense in the context\n * of sub-expression or append\n */\nexport type GenericKeywordNode = ASTv2.AppendContent | ASTv2.CallExpression;\n\nexport type KeywordNode =\n  | GenericKeywordNode\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock\n  | ASTv2.ElementModifier;\n\nexport function keyword<\n  K extends KeywordType,\n  D extends KeywordDelegate<KeywordMatches[K], unknown, Out>,\n  Out = unknown\n>(keyword: string, type: K, delegate: D): Keyword<K, Out> {\n  return new KeywordImpl(keyword, type, delegate as KeywordDelegate<KeywordMatch, unknown, Out>);\n}\n\nexport type PossibleKeyword = KeywordNode;\ntype OutFor<K extends Keyword | BlockKeyword> = K extends BlockKeyword<infer Out>\n  ? Out\n  : K extends Keyword<KeywordType, infer Out>\n  ? Out\n  : never;\n\nfunction getCalleeExpression(\n  node: KeywordNode | ASTv2.ExpressionNode\n): ASTv2.ExpressionNode | null {\n  switch (node.type) {\n    // This covers the inside of attributes and expressions, as well as the callee\n    // of call nodes\n    case 'Path':\n      return node;\n    case 'AppendContent':\n      return getCalleeExpression(node.value);\n    case 'Call':\n    case 'InvokeBlock':\n    case 'ElementModifier':\n      return node.callee;\n    default:\n      return null;\n  }\n}\n\nexport class Keywords<K extends KeywordType, KeywordList extends Keyword<K> = never>\n  implements Keyword<K, OutFor<KeywordList>> {\n  #keywords: Keyword[] = [];\n  #type: K;\n\n  constructor(type: K) {\n    this.#type = type;\n  }\n\n  kw<S extends string = string, Out = unknown>(\n    name: S,\n    delegate: KeywordDelegate<KeywordMatches[K], unknown, Out>\n  ): Keywords<K, KeywordList | Keyword<K, Out>> {\n    this.#keywords.push(keyword(name, this.#type, delegate));\n\n    return this;\n  }\n\n  translate(\n    node: KeywordCandidates[K],\n    state: NormalizationState\n  ): Result<OutFor<KeywordList>> | null {\n    for (let keyword of this.#keywords) {\n      let result = keyword.translate(node, state) as Result<OutFor<KeywordList>>;\n      if (result !== null) {\n        return result;\n      }\n    }\n\n    let path = getCalleeExpression(node);\n\n    if (path && path.type === 'Path' && path.ref.type === 'Free' && isKeyword(path.ref.name)) {\n      let { name } = path.ref;\n\n      let usedType = this.#type;\n      let validTypes = KEYWORDS_TYPES[name];\n\n      if (validTypes.indexOf(usedType) === -1) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${name}\\` keyword was used incorrectly. It was used as ${\n              typesToReadableName[usedType]\n            }, but its valid usages are:\\n\\n${generateTypesMessage(\n              name,\n              validTypes\n            )}\\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n    }\n\n    return null;\n  }\n}\n\nconst typesToReadableName = {\n  Append: 'an append statement',\n  Block: 'a block statement',\n  Call: 'a call expression',\n  Modifier: 'a modifier',\n};\n\nfunction generateTypesMessage(name: string, types: KeywordType[]): string {\n  return types\n    .map((type) => {\n      switch (type) {\n        case 'Append':\n          return `- As an append statement, as in: {{${name}}}`;\n        case 'Block':\n          return `- As a block statement, as in: {{#${name}}}{{/${name}}}`;\n        case 'Call':\n          return `- As an expression, as in: (${name})`;\n        case 'Modifier':\n          return `- As a modifier, as in: <div {{${name}}}></div>`;\n        default:\n          return exhausted(type);\n      }\n    })\n    .join('\\n\\n');\n}\n\n/**\n * This function builds keyword definitions for a particular type of AST node (`KeywordType`).\n *\n * You can build keyword definitions for:\n *\n * - `Expr`: A `SubExpression` or `PathExpression`\n * - `Block`: A `BlockStatement`\n *   - A `BlockStatement` is a keyword candidate if its head is a\n *     `PathExpression`\n * - `Append`: An `AppendStatement`\n *\n * A node is a keyword candidate if:\n *\n * - A `PathExpression` is a keyword candidate if it has no tail, and its\n *   head expression is a `LocalVarHead` or `FreeVarHead` whose name is\n *   the keyword's name.\n * - A `SubExpression`, `AppendStatement`, or `BlockStatement` is a keyword\n *   candidate if its head is a keyword candidate.\n *\n * The keyword infrastructure guarantees that:\n *\n * - If a node is not a keyword candidate, it is never passed to any keyword's\n *   `assert` method.\n * - If a node is not the `KeywordType` for a particular keyword, it will not\n *   be passed to the keyword's `assert` method.\n *\n * `Expr` keywords are used in expression positions and should return HIR\n * expressions. `Block` and `Append` keywords are used in statement\n * positions and should return HIR statements.\n *\n * A keyword definition has two parts:\n *\n * - `match`, which determines whether an AST node matches the keyword, and can\n *   optionally return some information extracted from the AST node.\n * - `translate`, which takes a matching AST node as well as the extracted\n *   information and returns an appropriate HIR instruction.\n *\n * # Example\n *\n * This keyword:\n *\n * - turns `(hello)` into `\"hello\"`\n *   - as long as `hello` is not in scope\n * - makes it an error to pass any arguments (such as `(hello world)`)\n *\n * ```ts\n * keywords('SubExpr').kw('hello', {\n *   assert(node: ExprKeywordNode): Result<void> | false {\n *     // we don't want to transform `hello` as a `PathExpression`\n *     if (node.type !== 'SubExpression') {\n *       return false;\n *     }\n *\n *     // node.head would be `LocalVarHead` if `hello` was in scope\n *     if (node.head.type !== 'FreeVarHead') {\n *       return false;\n *     }\n *\n *     if (node.params.length || node.hash) {\n *       return Err(generateSyntaxError(`(hello) does not take any arguments`), node.loc);\n *     } else {\n *       return Ok();\n *     }\n *   },\n *\n *   translate(node: ASTv2.SubExpression): hir.Expression {\n *     return ASTv2.builders.literal(\"hello\", node.loc)\n *   }\n * })\n * ```\n *\n * The keyword infrastructure checks to make sure that the node is the right\n * type before calling `assert`, so you only need to consider `SubExpression`\n * and `PathExpression` here. It also checks to make sure that the node passed\n * to `assert` has the keyword name in the right place.\n *\n * Note the important difference between returning `false` from `assert`,\n * which just means that the node didn't match, and returning `Err`, which\n * means that the node matched, but there was a keyword-specific syntax\n * error.\n */\nexport function keywords<K extends KeywordType>(type: K): Keywords<K> {\n  return new Keywords(type);\n}\n"],"sourceRoot":""} |
@@ -6,2 +6,8 @@ "use strict"; | ||
}); | ||
Object.defineProperty(exports, "defaultId", { | ||
enumerable: true, | ||
get: function () { | ||
return _compiler.defaultId; | ||
} | ||
}); | ||
Object.defineProperty(exports, "precompile", { | ||
@@ -75,2 +81,2 @@ enumerable: true, | ||
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3BhY2thZ2VzL0BnbGltbWVyL2NvbXBpbGVyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBOztBQUNBOztBQVlBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgcHJlY29tcGlsZSwgcHJlY29tcGlsZUpTT04sIFByZWNvbXBpbGVPcHRpb25zIH0gZnJvbSAnLi9saWIvY29tcGlsZXInO1xuZXhwb3J0IHtcbiAgUHJvZ3JhbVN5bWJvbHMsXG4gIGJ1aWxkU3RhdGVtZW50LFxuICBidWlsZFN0YXRlbWVudHMsXG4gIHMsXG4gIGMsXG4gIHVuaWNvZGUsXG4gIE5FV0xJTkUsXG59IGZyb20gJy4vbGliL2J1aWxkZXIvYnVpbGRlcic7XG5leHBvcnQgeyBCdWlsZGVyU3RhdGVtZW50LCBCdWlsZGVyIH0gZnJvbSAnLi9saWIvYnVpbGRlci9idWlsZGVyLWludGVyZmFjZSc7XG5cbi8vIGV4cG9ydGVkIG9ubHkgZm9yIHRlc3RzXG5leHBvcnQgeyBkZWZhdWx0IGFzIFdpcmVGb3JtYXREZWJ1Z2dlciB9IGZyb20gJy4vbGliL3dpcmUtZm9ybWF0LWRlYnVnJztcbiJdLCJzb3VyY2VSb290IjoiIn0= | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3BhY2thZ2VzL0BnbGltbWVyL2NvbXBpbGVyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBOztBQUNBOztBQVlBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgZGVmYXVsdElkLCBwcmVjb21waWxlLCBwcmVjb21waWxlSlNPTiwgUHJlY29tcGlsZU9wdGlvbnMgfSBmcm9tICcuL2xpYi9jb21waWxlcic7XG5leHBvcnQge1xuICBQcm9ncmFtU3ltYm9scyxcbiAgYnVpbGRTdGF0ZW1lbnQsXG4gIGJ1aWxkU3RhdGVtZW50cyxcbiAgcyxcbiAgYyxcbiAgdW5pY29kZSxcbiAgTkVXTElORSxcbn0gZnJvbSAnLi9saWIvYnVpbGRlci9idWlsZGVyJztcbmV4cG9ydCB7IEJ1aWxkZXJTdGF0ZW1lbnQsIEJ1aWxkZXIgfSBmcm9tICcuL2xpYi9idWlsZGVyL2J1aWxkZXItaW50ZXJmYWNlJztcblxuLy8gZXhwb3J0ZWQgb25seSBmb3IgdGVzdHNcbmV4cG9ydCB7IGRlZmF1bHQgYXMgV2lyZUZvcm1hdERlYnVnZ2VyIH0gZnJvbSAnLi9saWIvd2lyZS1mb3JtYXQtZGVidWcnO1xuIl0sInNvdXJjZVJvb3QiOiIifQ== |
@@ -8,2 +8,3 @@ "use strict"; | ||
exports.precompile = precompile; | ||
exports.defaultId = void 0; | ||
@@ -20,2 +21,30 @@ var _syntax = require("@glimmer/syntax"); | ||
var defaultId = function () { | ||
var req = typeof module === 'object' && typeof module.require === 'function' ? module.require : require; | ||
if (req) { | ||
try { | ||
var crypto = req('crypto'); | ||
var idFn = function idFn(src) { | ||
var hash = crypto.createHash('sha1'); | ||
hash.update(src, 'utf8'); // trim to 6 bytes of data (2^48 - 1) | ||
return hash.digest('base64').substring(0, 8); | ||
}; | ||
idFn('test'); | ||
return idFn; | ||
} catch (e) {} | ||
} | ||
return function idFn() { | ||
return null; | ||
}; | ||
}(); | ||
exports.defaultId = defaultId; | ||
var defaultOptions = { | ||
id: defaultId | ||
}; | ||
/* | ||
@@ -35,5 +64,6 @@ * Compile a string into a template javascript string. | ||
*/ | ||
function precompileJSON(string, options) { | ||
if (options === void 0) { | ||
options = {}; | ||
options = defaultOptions; | ||
} | ||
@@ -86,3 +116,3 @@ | ||
if (options === void 0) { | ||
options = {}; | ||
options = defaultOptions; | ||
} | ||
@@ -97,4 +127,6 @@ | ||
var moduleName = (_a = options.meta) === null || _a === void 0 ? void 0 : _a.moduleName; | ||
var idFn = options.id || defaultId; | ||
var blockJSON = JSON.stringify(block); | ||
var templateJSONObject = { | ||
id: idFn(JSON.stringify(options.meta) + blockJSON), | ||
block: blockJSON, | ||
@@ -124,2 +156,2 @@ moduleName: moduleName !== null && moduleName !== void 0 ? moduleName : '(unknown template module)', | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3BhY2thZ2VzL0BnbGltbWVyL2NvbXBpbGVyL2xpYi9jb21waWxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7OztBQU1BOztBQUNBOztBQUVBOztBQUNBOzs7O0FBRUE7Ozs7Ozs7Ozs7Ozs7O0FBY00sU0FBQSxjQUFBLENBQUEsTUFBQSxFQUFBLE9BQUEsRUFFMkI7QUFBQSxNQUEvQixPQUErQixLQUFBLEtBQUEsQ0FBQSxFQUFBO0FBQS9CLElBQUEsT0FBK0IsR0FGM0IsRUFFSjtBQUErQjs7OztBQUUvQixNQUFJLE1BQU0sR0FBRyxJQUFBLGNBQUEsQ0FBQSxNQUFBLEVBQWlCLENBQUEsRUFBQSxHQUFFLE9BQU8sQ0FBVCxJQUFBLE1BQUEsSUFBQSxJQUFjLEVBQUEsS0FBQSxLQUFkLENBQUEsR0FBYyxLQUFkLENBQUEsR0FBYyxFQUFBLENBQTVDLFVBQWEsQ0FBYjs7QUFGK0IsTUFBQSxVQUFBLEdBR1gsdUJBQVMsTUFBVCxFQUhXLE9BR1gsQ0FIVztBQUFBLE1BRzNCLEdBSDJCLEdBQUEsVUFBQSxDQUFBLENBQUEsQ0FBQTtBQUFBLE1BRzNCLE1BSDJCLEdBQUEsVUFBQSxDQUFBLENBQUEsQ0FBQTs7QUFJL0IsTUFBSSxLQUFLLEdBQUcsb0JBQUssTUFBTCxFQUFLLEdBQUwsRUFBaUIsQ0FBQSxFQUFBLEdBQUUsT0FBTyxDQUFULFVBQUEsTUFBQSxJQUFBLElBQW9CLEVBQUEsS0FBQSxLQUFwQixDQUFBLEdBQUEsRUFBQSxHQUFqQixLQUFBLEVBQUEsS0FBQSxDQUF1RCxVQUFELE9BQUMsRUFBVztBQUM1RSxXQUFPLG1CQUFQLE9BQU8sQ0FBUDtBQURGLEdBQVksQ0FBWjs7QUFJQSxNQUFBO0FBQUE7QUFBQSxJQUFzQjtBQUNwQix5QkFBQSxHQUFBLENBQUEsYUFBQSxFQUFBLEtBQUE7QUFDRDs7QUFFRCxNQUFJLEtBQUssQ0FBVCxJQUFBLEVBQWdCO0FBQ2QsV0FBTyxDQUFDLEtBQUssQ0FBTixLQUFBLEVBQVAsTUFBTyxDQUFQO0FBREYsR0FBQSxNQUVPO0FBQ0wsVUFBTSxLQUFLLENBQVgsTUFBQTtBQUNEO0VBR0g7QUFDQTs7O0FBQ0EsSUFBTSxpQkFBaUIsR0FBdkIsc0NBQUE7QUFFQTs7Ozs7Ozs7Ozs7Ozs7O0FBY00sU0FBQSxVQUFBLENBQUEsTUFBQSxFQUFBLE9BQUEsRUFBb0U7QUFBQSxNQUEvQixPQUErQixLQUFBLEtBQUEsQ0FBQSxFQUFBO0FBQS9CLElBQUEsT0FBK0IsR0FBcEUsRUFBcUM7QUFBK0I7Ozs7QUFBQSxNQUFBLGVBQUEsR0FDOUMsY0FBYyxDQUFBLE1BQUEsRUFEZ0MsT0FDaEMsQ0FEZ0M7QUFBQSxNQUNwRSxLQURvRSxHQUFBLGVBQUEsQ0FBQSxDQUFBLENBQUE7QUFBQSxNQUNwRSxVQURvRSxHQUFBLGVBQUEsQ0FBQSxDQUFBLENBQUE7O0FBR3hFLE1BQUksVUFBVSxHQUFBLENBQUEsRUFBQSxHQUFHLE9BQU8sQ0FBVixJQUFBLE1BQUEsSUFBQSxJQUFlLEVBQUEsS0FBQSxLQUFmLENBQUEsR0FBZSxLQUFmLENBQUEsR0FBZSxFQUFBLENBQTdCLFVBQUE7QUFDQSxNQUFJLFNBQVMsR0FBRyxJQUFJLENBQUosU0FBQSxDQUFoQixLQUFnQixDQUFoQjtBQUNBLE1BQUksa0JBQWtCLEdBQW9DO0FBQ3hELElBQUEsS0FBSyxFQURtRCxTQUFBO0FBRXhELElBQUEsVUFBVSxFQUFFLFVBQVUsS0FBVixJQUFBLElBQUEsVUFBVSxLQUFBLEtBQVYsQ0FBQSxHQUFBLFVBQUEsR0FGNEMsMkJBQUE7QUFHeEQ7QUFDQTtBQUNBLElBQUEsS0FBSyxFQUxtRCxpQkFBQTtBQU14RCxJQUFBLFlBQVksRUFBQSxDQUFBLEVBQUEsR0FBRSxPQUFPLENBQVQsVUFBQSxNQUFBLElBQUEsSUFBb0IsRUFBQSxLQUFBLEtBQXBCLENBQUEsR0FBQSxFQUFBLEdBQXdCO0FBTm9CLEdBQTFEOztBQVNBLE1BQUksQ0FBQyxPQUFPLENBQVosVUFBQSxFQUF5QjtBQUN2QixXQUFPLGtCQUFrQixDQUF6QixLQUFBO0FBZnNFLEdBQUEsQ0FrQnhFOzs7QUFDQSxNQUFJLFdBQVcsR0FBRyxJQUFJLENBQUosU0FBQSxDQUFsQixrQkFBa0IsQ0FBbEI7O0FBRUEsTUFBSSxPQUFPLENBQVAsVUFBQSxJQUFzQixVQUFVLENBQVYsTUFBQSxHQUExQixDQUFBLEVBQWlEO0FBQy9DLFFBQUksT0FBTyxHQUFBLFVBQVcsVUFBVSxDQUFWLElBQUEsQ0FBdEIsR0FBc0IsQ0FBWCxHQUFYLEdBQUE7QUFFQSxJQUFBLFdBQVcsR0FBRyxXQUFXLENBQVgsT0FBQSxDQUFBLE9BQUEsaUJBQUEsR0FBQSxJQUFBLEVBQWQsT0FBYyxDQUFkO0FBSEYsR0FBQSxNQUlPO0FBQ0wsSUFBQSxXQUFXLEdBQUcsV0FBVyxDQUFYLE9BQUEsQ0FBQSxPQUFBLGlCQUFBLEdBQUEsSUFBQSxFQUFkLE1BQWMsQ0FBZDtBQUNEOztBQUVELFNBQUEsV0FBQTtBQUNEIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgU2VyaWFsaXplZFRlbXBsYXRlQmxvY2ssXG4gIFNlcmlhbGl6ZWRUZW1wbGF0ZVdpdGhMYXp5QmxvY2ssXG4gIFRlbXBsYXRlSmF2YXNjcmlwdCxcbn0gZnJvbSAnQGdsaW1tZXIvaW50ZXJmYWNlcyc7XG5pbXBvcnQgeyBMT0NBTF9TSE9VTERfTE9HIH0gZnJvbSAnQGdsaW1tZXIvbG9jYWwtZGVidWctZmxhZ3MnO1xuaW1wb3J0IHsgbm9ybWFsaXplLCBQcmVjb21waWxlT3B0aW9ucywgU291cmNlIH0gZnJvbSAnQGdsaW1tZXIvc3ludGF4JztcbmltcG9ydCB7IExPQ0FMX0xPR0dFUiB9IGZyb20gJ0BnbGltbWVyL3V0aWwnO1xuXG5pbXBvcnQgcGFzczAgZnJvbSAnLi9wYXNzZXMvMS1ub3JtYWxpemF0aW9uL2luZGV4JztcbmltcG9ydCB7IHZpc2l0IGFzIHBhc3MyIH0gZnJvbSAnLi9wYXNzZXMvMi1lbmNvZGluZy9pbmRleCc7XG5cbi8qXG4gKiBDb21waWxlIGEgc3RyaW5nIGludG8gYSB0ZW1wbGF0ZSBqYXZhc2NyaXB0IHN0cmluZy5cbiAqXG4gKiBFeGFtcGxlIHVzYWdlOlxuICogICAgIGltcG9ydCB7IHByZWNvbXBpbGUgfSBmcm9tICdAZ2xpbW1lci9jb21waWxlcic7XG4gKiAgICAgaW1wb3J0IHsgdGVtcGxhdGVGYWN0b3J5IH0gZnJvbSAnZ2xpbW1lci1ydW50aW1lJztcbiAqICAgICBsZXQgdGVtcGxhdGVKcyA9IHByZWNvbXBpbGUoXCJIb3dkeSB7e25hbWV9fVwiKTtcbiAqICAgICBsZXQgZmFjdG9yeSA9IHRlbXBsYXRlRmFjdG9yeShuZXcgRnVuY3Rpb24oXCJyZXR1cm4gXCIgKyB0ZW1wbGF0ZUpzKSgpKTtcbiAqICAgICBsZXQgdGVtcGxhdGUgPSBmYWN0b3J5LmNyZWF0ZShlbnYpO1xuICpcbiAqIEBtZXRob2QgcHJlY29tcGlsZVxuICogQHBhcmFtIHtzdHJpbmd9IHN0cmluZyBhIEdsaW1tZXIgdGVtcGxhdGUgc3RyaW5nXG4gKiBAcmV0dXJuIHtzdHJpbmd9IGEgdGVtcGxhdGUgamF2YXNjcmlwdCBzdHJpbmdcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHByZWNvbXBpbGVKU09OKFxuICBzdHJpbmc6IHN0cmluZyxcbiAgb3B0aW9uczogUHJlY29tcGlsZU9wdGlvbnMgPSB7fVxuKTogW2Jsb2NrOiBTZXJpYWxpemVkVGVtcGxhdGVCbG9jaywgdXNlZExvY2Fsczogc3RyaW5nW11dIHtcbiAgbGV0IHNvdXJjZSA9IG5ldyBTb3VyY2Uoc3RyaW5nLCBvcHRpb25zLm1ldGE/Lm1vZHVsZU5hbWUpO1xuICBsZXQgW2FzdCwgbG9jYWxzXSA9IG5vcm1hbGl6ZShzb3VyY2UsIG9wdGlvbnMpO1xuICBsZXQgYmxvY2sgPSBwYXNzMChzb3VyY2UsIGFzdCwgb3B0aW9ucy5zdHJpY3RNb2RlID8/IGZhbHNlKS5tYXBPaygocGFzczJJbikgPT4ge1xuICAgIHJldHVybiBwYXNzMihwYXNzMkluKTtcbiAgfSk7XG5cbiAgaWYgKExPQ0FMX1NIT1VMRF9MT0cpIHtcbiAgICBMT0NBTF9MT0dHRVIubG9nKGBUZW1wbGF0ZSAtPmAsIGJsb2NrKTtcbiAgfVxuXG4gIGlmIChibG9jay5pc09rKSB7XG4gICAgcmV0dXJuIFtibG9jay52YWx1ZSwgbG9jYWxzXTtcbiAgfSBlbHNlIHtcbiAgICB0aHJvdyBibG9jay5yZWFzb247XG4gIH1cbn1cblxuLy8gVVVJRCB1c2VkIGFzIGEgdW5pcXVlIHBsYWNlaG9sZGVyIGZvciBwbGFjaW5nIGEgc25pcHBldCBvZiBKUyBjb2RlIGludG9cbi8vIHRoZSBvdGhlcndpc2UgSlNPTiBzdHJpbmdpZmllZCB2YWx1ZSBiZWxvdy5cbmNvbnN0IFNDT1BFX1BMQUNFSE9MREVSID0gJzc5NmQyNGU2LTI0NTAtNGZiMC04Y2RmLWI2NTYzOGI1ZWY3MCc7XG5cbi8qXG4gKiBDb21waWxlIGEgc3RyaW5nIGludG8gYSB0ZW1wbGF0ZSBqYXZhc2NyaXB0IHN0cmluZy5cbiAqXG4gKiBFeGFtcGxlIHVzYWdlOlxuICogICAgIGltcG9ydCB7IHByZWNvbXBpbGUgfSBmcm9tICdAZ2xpbW1lci9jb21waWxlcic7XG4gKiAgICAgaW1wb3J0IHsgdGVtcGxhdGVGYWN0b3J5IH0gZnJvbSAnZ2xpbW1lci1ydW50aW1lJztcbiAqICAgICBsZXQgdGVtcGxhdGVKcyA9IHByZWNvbXBpbGUoXCJIb3dkeSB7e25hbWV9fVwiKTtcbiAqICAgICBsZXQgZmFjdG9yeSA9IHRlbXBsYXRlRmFjdG9yeShuZXcgRnVuY3Rpb24oXCJyZXR1cm4gXCIgKyB0ZW1wbGF0ZUpzKSgpKTtcbiAqICAgICBsZXQgdGVtcGxhdGUgPSBmYWN0b3J5LmNyZWF0ZShlbnYpO1xuICpcbiAqIEBtZXRob2QgcHJlY29tcGlsZVxuICogQHBhcmFtIHtzdHJpbmd9IHN0cmluZyBhIEdsaW1tZXIgdGVtcGxhdGUgc3RyaW5nXG4gKiBAcmV0dXJuIHtzdHJpbmd9IGEgdGVtcGxhdGUgamF2YXNjcmlwdCBzdHJpbmdcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHByZWNvbXBpbGUoc291cmNlOiBzdHJpbmcsIG9wdGlvbnM6IFByZWNvbXBpbGVPcHRpb25zID0ge30pOiBUZW1wbGF0ZUphdmFzY3JpcHQge1xuICBsZXQgW2Jsb2NrLCB1c2VkTG9jYWxzXSA9IHByZWNvbXBpbGVKU09OKHNvdXJjZSwgb3B0aW9ucyk7XG5cbiAgbGV0IG1vZHVsZU5hbWUgPSBvcHRpb25zLm1ldGE/Lm1vZHVsZU5hbWU7XG4gIGxldCBibG9ja0pTT04gPSBKU09OLnN0cmluZ2lmeShibG9jayk7XG4gIGxldCB0ZW1wbGF0ZUpTT05PYmplY3Q6IFNlcmlhbGl6ZWRUZW1wbGF0ZVdpdGhMYXp5QmxvY2sgPSB7XG4gICAgYmxvY2s6IGJsb2NrSlNPTixcbiAgICBtb2R1bGVOYW1lOiBtb2R1bGVOYW1lID8/ICcodW5rbm93biB0ZW1wbGF0ZSBtb2R1bGUpJyxcbiAgICAvLyBseWluZyB0byB0aGUgdHlwZSBjaGVja2VyIGhlcmUgYmVjYXVzZSB3ZSdyZSBnb2luZyB0b1xuICAgIC8vIHJlcGxhY2UgaXQganVzdCBiZWxvdywgYWZ0ZXIgc3RyaW5naWZpY2F0aW9uXG4gICAgc2NvcGU6IChTQ09QRV9QTEFDRUhPTERFUiBhcyB1bmtub3duKSBhcyBudWxsLFxuICAgIGlzU3RyaWN0TW9kZTogb3B0aW9ucy5zdHJpY3RNb2RlID8/IGZhbHNlLFxuICB9O1xuXG4gIGlmICghb3B0aW9ucy5zdHJpY3RNb2RlKSB7XG4gICAgZGVsZXRlIHRlbXBsYXRlSlNPTk9iamVjdC5zY29wZTtcbiAgfVxuXG4gIC8vIEpTT04gaXMgamF2YXNjcmlwdFxuICBsZXQgc3RyaW5naWZpZWQgPSBKU09OLnN0cmluZ2lmeSh0ZW1wbGF0ZUpTT05PYmplY3QpO1xuXG4gIGlmIChvcHRpb25zLnN0cmljdE1vZGUgJiYgdXNlZExvY2Fscy5sZW5ndGggPiAwKSB7XG4gICAgbGV0IHNjb3BlRm4gPSBgKCk9Plske3VzZWRMb2NhbHMuam9pbignLCcpfV1gO1xuXG4gICAgc3RyaW5naWZpZWQgPSBzdHJpbmdpZmllZC5yZXBsYWNlKGBcIiR7U0NPUEVfUExBQ0VIT0xERVJ9XCJgLCBzY29wZUZuKTtcbiAgfSBlbHNlIHtcbiAgICBzdHJpbmdpZmllZCA9IHN0cmluZ2lmaWVkLnJlcGxhY2UoYFwiJHtTQ09QRV9QTEFDRUhPTERFUn1cImAsICdudWxsJyk7XG4gIH1cblxuICByZXR1cm4gc3RyaW5naWZpZWQ7XG59XG5cbmV4cG9ydCB7IFByZWNvbXBpbGVPcHRpb25zIH07XG4iXSwic291cmNlUm9vdCI6IiJ9 | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../packages/@glimmer/compiler/lib/compiler.ts"],"names":[],"mappings":";;;;;;;;;AAMA;;AACA;;AAEA;;AACA;;;;AAcO,IAAM,SAAS,GAAkB,YAAK;AAC3C,MAAI,GAAG,GACL,OAAA,MAAA,KAAA,QAAA,IAA8B,OAAO,MAAM,CAAb,OAAA,KAA9B,UAAA,GAAqE,MAAM,CAA3E,OAAA,GADF,OAAA;;AAGA,MAAA,GAAA,EAAS;AACP,QAAI;AACF,UAAM,MAAM,GAAG,GAAG,CAAlB,QAAkB,CAAlB;;AAEA,UAAI,IAAI,GAAkB,SAAtB,IAAsB,CAAD,GAAC,EAAO;AAC/B,YAAI,IAAI,GAAG,MAAM,CAAN,UAAA,CAAX,MAAW,CAAX;AACA,QAAA,IAAI,CAAJ,MAAA,CAAA,GAAA,EAF+B,MAE/B,EAF+B,CAG/B;;AACA,eAAO,IAAI,CAAJ,MAAA,CAAA,QAAA,EAAA,SAAA,CAAA,CAAA,EAAP,CAAO,CAAP;AAJF,OAAA;;AAOA,MAAA,IAAI,CAAJ,MAAI,CAAJ;AAEA,aAAA,IAAA;AAZF,KAAA,CAaE,OAAA,CAAA,EAAU,CAAE;AACf;;AAED,SAAO,SAAA,IAAA,GAAa;AAClB,WAAA,IAAA;AADF,GAAA;AArBK,CAAiC,EAAjC;;;AA0BP,IAAM,cAAc,GAAsB;AACxC,EAAA,EAAE,EAAE;AADoC,CAA1C;AAIA;;;;;;;;;;;;;;;AAcM,SAAA,cAAA,CAAA,MAAA,EAAA,OAAA,EAEuC;AAAA,MAA3C,OAA2C,KAAA,KAAA,CAAA,EAAA;AAA3C,IAAA,OAA2C,GAFvC,cAEJ;AAA2C;;;;AAE3C,MAAI,MAAM,GAAG,IAAA,cAAA,CAAA,MAAA,EAAiB,CAAA,EAAA,GAAE,OAAO,CAAT,IAAA,MAAA,IAAA,IAAc,EAAA,KAAA,KAAd,CAAA,GAAc,KAAd,CAAA,GAAc,EAAA,CAA5C,UAAa,CAAb;;AAF2C,MAAA,UAAA,GAGvB,uBAAS,MAAT,EAHuB,OAGvB,CAHuB;AAAA,MAGvC,GAHuC,GAAA,UAAA,CAAA,CAAA,CAAA;AAAA,MAGvC,MAHuC,GAAA,UAAA,CAAA,CAAA,CAAA;;AAI3C,MAAI,KAAK,GAAG,oBAAK,MAAL,EAAK,GAAL,EAAiB,CAAA,EAAA,GAAE,OAAO,CAAT,UAAA,MAAA,IAAA,IAAoB,EAAA,KAAA,KAApB,CAAA,GAAA,EAAA,GAAjB,KAAA,EAAA,KAAA,CAAuD,UAAD,OAAC,EAAW;AAC5E,WAAO,mBAAP,OAAO,CAAP;AADF,GAAY,CAAZ;;AAIA,MAAA;AAAA;AAAA,IAAsB;AACpB,yBAAA,GAAA,CAAA,aAAA,EAAA,KAAA;AACD;;AAED,MAAI,KAAK,CAAT,IAAA,EAAgB;AACd,WAAO,CAAC,KAAK,CAAN,KAAA,EAAP,MAAO,CAAP;AADF,GAAA,MAEO;AACL,UAAM,KAAK,CAAX,MAAA;AACD;EAGH;AACA;;;AACA,IAAM,iBAAiB,GAAvB,sCAAA;AAEA;;;;;;;;;;;;;;;AAcM,SAAA,UAAA,CAAA,MAAA,EAAA,OAAA,EAEuC;AAAA,MAA3C,OAA2C,KAAA,KAAA,CAAA,EAAA;AAA3C,IAAA,OAA2C,GAFvC,cAEJ;AAA2C;;;;AAAA,MAAA,eAAA,GAEjB,cAAc,CAAA,MAAA,EAFG,OAEH,CAFG;AAAA,MAEvC,KAFuC,GAAA,eAAA,CAAA,CAAA,CAAA;AAAA,MAEvC,UAFuC,GAAA,eAAA,CAAA,CAAA,CAAA;;AAI3C,MAAI,UAAU,GAAA,CAAA,EAAA,GAAG,OAAO,CAAV,IAAA,MAAA,IAAA,IAAe,EAAA,KAAA,KAAf,CAAA,GAAe,KAAf,CAAA,GAAe,EAAA,CAA7B,UAAA;AACA,MAAI,IAAI,GAAG,OAAO,CAAP,EAAA,IAAX,SAAA;AACA,MAAI,SAAS,GAAG,IAAI,CAAJ,SAAA,CAAhB,KAAgB,CAAhB;AACA,MAAI,kBAAkB,GAAoC;AACxD,IAAA,EAAE,EAAE,IAAI,CAAC,IAAI,CAAJ,SAAA,CAAe,OAAO,CAAtB,IAAA,IAD+C,SAChD,CADgD;AAExD,IAAA,KAAK,EAFmD,SAAA;AAGxD,IAAA,UAAU,EAAE,UAAU,KAAV,IAAA,IAAA,UAAU,KAAA,KAAV,CAAA,GAAA,UAAA,GAH4C,2BAAA;AAIxD;AACA;AACA,IAAA,KAAK,EANmD,iBAAA;AAOxD,IAAA,YAAY,EAAA,CAAA,EAAA,GAAE,OAAO,CAAT,UAAA,MAAA,IAAA,IAAoB,EAAA,KAAA,KAApB,CAAA,GAAA,EAAA,GAAwB;AAPoB,GAA1D;;AAUA,MAAI,CAAC,OAAO,CAAZ,UAAA,EAAyB;AACvB,WAAO,kBAAkB,CAAzB,KAAA;AAlByC,GAAA,CAqB3C;;;AACA,MAAI,WAAW,GAAG,IAAI,CAAJ,SAAA,CAAlB,kBAAkB,CAAlB;;AAEA,MAAI,OAAO,CAAP,UAAA,IAAsB,UAAU,CAAV,MAAA,GAA1B,CAAA,EAAiD;AAC/C,QAAI,OAAO,GAAA,UAAW,UAAU,CAAV,IAAA,CAAtB,GAAsB,CAAX,GAAX,GAAA;AAEA,IAAA,WAAW,GAAG,WAAW,CAAX,OAAA,CAAA,OAAA,iBAAA,GAAA,IAAA,EAAd,OAAc,CAAd;AAHF,GAAA,MAIO;AACL,IAAA,WAAW,GAAG,WAAW,CAAX,OAAA,CAAA,OAAA,iBAAA,GAAA,IAAA,EAAd,MAAc,CAAd;AACD;;AAED,SAAA,WAAA;AACD","sourcesContent":["import {\n  SerializedTemplateBlock,\n  SerializedTemplateWithLazyBlock,\n  TemplateJavascript,\n} from '@glimmer/interfaces';\nimport { LOCAL_SHOULD_LOG } from '@glimmer/local-debug-flags';\nimport { normalize, PrecompileOptions, Source, TemplateIdFn } from '@glimmer/syntax';\nimport { LOCAL_LOGGER } from '@glimmer/util';\n\nimport pass0 from './passes/1-normalization/index';\nimport { visit as pass2 } from './passes/2-encoding/index';\n\ndeclare function require(id: 'crypto'): Crypto;\ndeclare function require(id: string): unknown;\n\ninterface Crypto {\n  createHash(\n    alg: 'sha1'\n  ): {\n    update(src: string, encoding: 'utf8'): void;\n    digest(encoding: 'base64'): string;\n  };\n}\n\nexport const defaultId: TemplateIdFn = (() => {\n  let req: typeof require | undefined =\n    typeof module === 'object' && typeof module.require === 'function' ? module.require : require;\n\n  if (req) {\n    try {\n      const crypto = req('crypto');\n\n      let idFn: TemplateIdFn = (src) => {\n        let hash = crypto.createHash('sha1');\n        hash.update(src, 'utf8');\n        // trim to 6 bytes of data (2^48 - 1)\n        return hash.digest('base64').substring(0, 8);\n      };\n\n      idFn('test');\n\n      return idFn;\n    } catch (e) {}\n  }\n\n  return function idFn() {\n    return null;\n  };\n})();\n\nconst defaultOptions: PrecompileOptions = {\n  id: defaultId,\n};\n\n/*\n * Compile a string into a template javascript string.\n *\n * Example usage:\n *     import { precompile } from '@glimmer/compiler';\n *     import { templateFactory } from 'glimmer-runtime';\n *     let templateJs = precompile(\"Howdy {{name}}\");\n *     let factory = templateFactory(new Function(\"return \" + templateJs)());\n *     let template = factory.create(env);\n *\n * @method precompile\n * @param {string} string a Glimmer template string\n * @return {string} a template javascript string\n */\nexport function precompileJSON(\n  string: string,\n  options: PrecompileOptions = defaultOptions\n): [block: SerializedTemplateBlock, usedLocals: string[]] {\n  let source = new Source(string, options.meta?.moduleName);\n  let [ast, locals] = normalize(source, options);\n  let block = pass0(source, ast, options.strictMode ?? false).mapOk((pass2In) => {\n    return pass2(pass2In);\n  });\n\n  if (LOCAL_SHOULD_LOG) {\n    LOCAL_LOGGER.log(`Template ->`, block);\n  }\n\n  if (block.isOk) {\n    return [block.value, locals];\n  } else {\n    throw block.reason;\n  }\n}\n\n// UUID used as a unique placeholder for placing a snippet of JS code into\n// the otherwise JSON stringified value below.\nconst SCOPE_PLACEHOLDER = '796d24e6-2450-4fb0-8cdf-b65638b5ef70';\n\n/*\n * Compile a string into a template javascript string.\n *\n * Example usage:\n *     import { precompile } from '@glimmer/compiler';\n *     import { templateFactory } from 'glimmer-runtime';\n *     let templateJs = precompile(\"Howdy {{name}}\");\n *     let factory = templateFactory(new Function(\"return \" + templateJs)());\n *     let template = factory.create(env);\n *\n * @method precompile\n * @param {string} string a Glimmer template string\n * @return {string} a template javascript string\n */\nexport function precompile(\n  source: string,\n  options: PrecompileOptions = defaultOptions\n): TemplateJavascript {\n  let [block, usedLocals] = precompileJSON(source, options);\n\n  let moduleName = options.meta?.moduleName;\n  let idFn = options.id || defaultId;\n  let blockJSON = JSON.stringify(block);\n  let templateJSONObject: SerializedTemplateWithLazyBlock = {\n    id: idFn(JSON.stringify(options.meta) + blockJSON),\n    block: blockJSON,\n    moduleName: moduleName ?? '(unknown template module)',\n    // lying to the type checker here because we're going to\n    // replace it just below, after stringification\n    scope: (SCOPE_PLACEHOLDER as unknown) as null,\n    isStrictMode: options.strictMode ?? false,\n  };\n\n  if (!options.strictMode) {\n    delete templateJSONObject.scope;\n  }\n\n  // JSON is javascript\n  let stringified = JSON.stringify(templateJSONObject);\n\n  if (options.strictMode && usedLocals.length > 0) {\n    let scopeFn = `()=>[${usedLocals.join(',')}]`;\n\n    stringified = stringified.replace(`\"${SCOPE_PLACEHOLDER}\"`, scopeFn);\n  } else {\n    stringified = stringified.replace(`\"${SCOPE_PLACEHOLDER}\"`, 'null');\n  }\n\n  return stringified;\n}\n\nexport { PrecompileOptions };\n"],"sourceRoot":""} |
@@ -100,5 +100,5 @@ "use strict"; | ||
var path = getPathExpression(node); | ||
var path = getCalleeExpression(node); | ||
if (path !== null && path.ref.type === 'Free') { | ||
if (path !== null && path.type === 'Path' && path.ref.type === 'Free') { | ||
if (path.tail.length > 0) { | ||
@@ -122,5 +122,5 @@ if (path.ref.resolution.serialize() === 'Loose') { | ||
if (this.match(node)) { | ||
var path = getPathExpression(node); | ||
var path = getCalleeExpression(node); | ||
if (path !== null && path.tail.length > 0) { | ||
if (path !== null && path.type === 'Path' && path.tail.length > 0) { | ||
return (0, _result.Err)((0, _syntax.generateSyntaxError)("The `" + this.keyword + "` keyword was used incorrectly. It was used as `" + path.loc.asString() + "`, but it cannot be used with additional path segments. \n\nError caused by", node.loc)); | ||
@@ -156,3 +156,3 @@ } | ||
function getPathExpression(node) { | ||
function getCalleeExpression(node) { | ||
switch (node.type) { | ||
@@ -165,3 +165,3 @@ // This covers the inside of attributes and expressions, as well as the callee | ||
case 'AppendContent': | ||
return getPathExpression(node.value); | ||
return getCalleeExpression(node.value); | ||
@@ -171,3 +171,3 @@ case 'Call': | ||
case 'ElementModifier': | ||
return getPathExpression(node.callee); | ||
return node.callee; | ||
@@ -207,5 +207,5 @@ default: | ||
var path = getPathExpression(node); | ||
var path = getCalleeExpression(node); | ||
if (path && path.ref.type === 'Free' && (0, _syntax.isKeyword)(path.ref.name)) { | ||
if (path && path.type === 'Path' && path.ref.type === 'Free' && (0, _syntax.isKeyword)(path.ref.name)) { | ||
var name = path.ref.name; | ||
@@ -343,2 +343,2 @@ | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../../../packages/@glimmer/compiler/lib/passes/1-normalization/keywords/impl.ts"],"names":[],"mappings":";;;;;;;;;AAAA;;AAOA;;AAEA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;IAgBA,W;AAQE,WAAA,WAAA,CAAA,OAAA,EAAA,IAAA,EAAA,QAAA,EAGkE;AAFtD,SAAA,OAAA,GAAA,OAAA;AAEF,SAAA,QAAA,GAAA,QAAA;AAER,QAAI,KAAK,GAAG,IAAZ,GAAY,EAAZ;;AACA,SAAA,IAAA,SAAA,GAAA,+BAAA,CAAqB,aAAa,CAAlC,IAAkC,CAAlC,CAAA,EAAA,KAAA,EAAA,CAAA,CAAA,KAAA,GAAA,SAAA,EAAA,EAAA,IAAA,GAA0C;AAAA,UAA1C,QAA0C,GAAA,KAAA,CAAA,KAAA;AACxC,MAAA,KAAK,CAAL,GAAA,CAAA,QAAA;AACD;;AAED,SAAA,KAAA,GAAA,KAAA;AACD;;;;SAES,K,GAAA,SAAA,KAAA,CAAA,IAAA,EAAgC;AACxC,QAAI,CAAC,KAAA,KAAA,CAAA,GAAA,CAAe,IAAI,CAAxB,IAAK,CAAL,EAAgC;AAC9B,aAAA,KAAA;AACD;;AAED,QAAI,IAAI,GAAG,iBAAiB,CAA5B,IAA4B,CAA5B;;AAEA,QAAI,IAAI,KAAJ,IAAA,IAAiB,IAAI,CAAJ,GAAA,CAAA,IAAA,KAArB,MAAA,EAA+C;AAC7C,UAAI,IAAI,CAAJ,IAAA,CAAA,MAAA,GAAJ,CAAA,EAA0B;AACxB,YAAI,IAAI,CAAJ,GAAA,CAAA,UAAA,CAAA,SAAA,OAAJ,OAAA,EAAiD;AAC/C;AACA;AACA,iBAAA,KAAA;AACD;AACF;;AAED,aAAO,IAAI,CAAJ,GAAA,CAAA,IAAA,KAAkB,KAAzB,OAAA;AATF,KAAA,MAUO;AACL,aAAA,KAAA;AACD;;;SAGH,S,GAAA,SAAA,SAAA,CAAA,IAAA,EAAA,KAAA,EAA4D;AAAA,QAAA,KAAA,GAAA,IAAA;;AAC1D,QAAI,KAAA,KAAA,CAAJ,IAAI,CAAJ,EAAsB;AACpB,UAAI,IAAI,GAAG,iBAAiB,CAA5B,IAA4B,CAA5B;;AAEA,UAAI,IAAI,KAAJ,IAAA,IAAiB,IAAI,CAAJ,IAAA,CAAA,MAAA,GAArB,CAAA,EAA2C;AACzC,eAAO,iBACL,iCAAmB,UAEf,KAFe,OAAA,GAAA,kDAAA,GAGoC,IAAI,CAAJ,GAAA,CAHpC,QAGoC,EAHpC,GAAA,6EAAnB,EAIE,IAAI,CALR,GACE,CADK,CAAP;AAQD;;AAED,UAAI,KAAK,GAAG,KAAA,QAAA,CAAA,MAAA,CAAA,IAAA,EAAZ,KAAY,CAAZ;AACA,aAAO,KAAK,CAAL,OAAA,CAAe,UAAD,KAAC,EAAD;AAAA,eAAW,KAAA,CAAA,QAAA,CAAA,SAAA,CAAwB;AAAE,UAAA,IAAF,EAAA,IAAA;AAAQ,UAAA,KAAA,EAAA;AAAR,SAAxB,EAAhC,KAAgC,CAAX;AAArB,OAAO,CAAP;AAfF,KAAA,MAgBO;AACL,aAAA,IAAA;AACD;;;;;;AAUE,IAAM,aAAa,GAAG;AAC3B,EAAA,IAAI,EAAE,CADqB,MACrB,CADqB;AAE3B,EAAA,KAAK,EAAE,CAFoB,aAEpB,CAFoB;AAG3B,EAAA,MAAM,EAAE,CAHmB,eAGnB,CAHmB;AAI3B,EAAA,QAAQ,EAAE,CAAA,iBAAA;AAJiB,CAAtB;;;AAqCD,SAAA,OAAA,CAAA,OAAA,EAAA,IAAA,EAAA,QAAA,EAIiC;AACrC,SAAO,IAAA,WAAA,CAAA,OAAA,EAAA,IAAA,EAAP,QAAO,CAAP;AACD;;AASD,SAAA,iBAAA,CAAA,IAAA,EAAmE;AACjE,UAAQ,IAAI,CAAZ,IAAA;AACE;AACA;AACA,SAAA,MAAA;AACE,aAAA,IAAA;;AACF,SAAA,eAAA;AACE,aAAO,iBAAiB,CAAC,IAAI,CAA7B,KAAwB,CAAxB;;AACF,SAAA,MAAA;AACA,SAAA,aAAA;AACA,SAAA,iBAAA;AACE,aAAO,iBAAiB,CAAC,IAAI,CAA7B,MAAwB,CAAxB;;AACF;AACE,aAAA,IAAA;AAZJ;AAcD;;AAED,IAAM,QAAN,GAAA,aAAA,YAAA;AAKE,WAAA,QAAA,CAAA,IAAA,EAAmB;AAHnB,IAAA,SAAA,CAAA,GAAA,CAAA,IAAA,EAAA,EAAA;;AACA,IAAA,KAAA,CAAA,GAAA,CAAA,IAAA,EAAA,KAAA,CAAA;;AAGE,IAAA,sBAAA,CAAA,IAAA,EAAA,KAAA,EAAA,IAAA,CAAA;AACD;;AAPH,MAAA,OAAA,GAAA,QAAA,CAAA,SAAA;;AAAA,EAAA,OAAA,CAAA,EAAA,GASE,SAAA,EAAA,CAAA,IAAA,EAAA,QAAA,EAE4D;AAE1D,IAAA,sBAAA,CAAA,IAAA,EAAA,SAAA,CAAA,CAAA,IAAA,CAAoB,OAAO,CAAA,IAAA,EAAK,sBAAA,CAAA,IAAA,EAAL,KAAK,CAAL,EAA3B,QAA2B,CAA3B;;AAEA,WAAA,IAAA;AAfJ,GAAA;;AAAA,EAAA,OAAA,CAAA,SAAA,GAkBE,SAAA,SAAA,CAAA,IAAA,EAAA,KAAA,EAE2B;AAEzB,SAAA,IAAA,UAAA,GAAA,+BAAA,CAAgB,sBAAA,CAAA,IAAA,EAAhB,SAAgB,CAAhB,CAAA,EAAA,MAAA,EAAA,CAAA,CAAA,MAAA,GAAA,UAAA,EAAA,EAAA,IAAA,GAAoC;AAAA,UAApC,QAAoC,GAAA,MAAA,CAAA,KAAA;;AAClC,UAAI,MAAM,GAAG,QAAO,CAAP,SAAA,CAAA,IAAA,EAAb,KAAa,CAAb;;AACA,UAAI,MAAM,KAAV,IAAA,EAAqB;AACnB,eAAA,MAAA;AACD;AACF;;AAED,QAAI,IAAI,GAAG,iBAAiB,CAA5B,IAA4B,CAA5B;;AAEA,QAAI,IAAI,IAAI,IAAI,CAAJ,GAAA,CAAA,IAAA,KAAR,MAAA,IAAoC,uBAAU,IAAI,CAAJ,GAAA,CAAlD,IAAwC,CAAxC,EAAkE;AAAA,UAC1D,IAD0D,GACjD,IAAI,CAD6C,GACjD,CADiD,IAAA;;AAGhE,UAAI,QAAQ,GAAA,sBAAA,CAAA,IAAA,EAAZ,KAAY,CAAZ;;AACA,UAAI,UAAU,GAAG,uBAAjB,IAAiB,CAAjB;;AAEA,UAAI,UAAU,CAAV,OAAA,CAAA,QAAA,MAAiC,CAArC,CAAA,EAAyC;AACvC,eAAO,iBACL,iCAAmB,UAAA,IAAA,GAAA,iDAAA,GAEf,mBAAmB,CAFJ,QAEI,CAFJ,GAAA,iCAAA,GAGiB,oBAAoB,CAAA,IAAA,EAHrC,UAGqC,CAHrC,GAAA,qBAAnB,EAOE,IAAI,CARR,GACE,CADK,CAAP;AAWD;AACF;;AAED,WAAA,IAAA;AApDJ,GAAA;;AAAA,SAAA,QAAA;AAAA,CAAA,EAAA;;;;AAwDA,IAAM,mBAAmB,GAAG;AAC1B,EAAA,MAAM,EADoB,qBAAA;AAE1B,EAAA,KAAK,EAFqB,mBAAA;AAG1B,EAAA,IAAI,EAHsB,mBAAA;AAI1B,EAAA,QAAQ,EAAE;AAJgB,CAA5B;;AAOA,SAAA,oBAAA,CAAA,IAAA,EAAA,KAAA,EAAgE;AAC9D,SAAO,KAAK,CAAL,GAAA,CACC,UAAD,IAAC,EAAQ;AACZ,YAAA,IAAA;AACE,WAAA,QAAA;AACE,eAAA,wCAAA,IAAA,GAAA,IAAA;;AACF,WAAA,OAAA;AACE,eAAA,uCAAA,IAAA,GAAA,OAAA,GAAA,IAAA,GAAA,IAAA;;AACF,WAAA,MAAA;AACE,eAAA,iCAAA,IAAA,GAAA,GAAA;;AACF,WAAA,UAAA;AACE,eAAA,oCAAA,IAAA,GAAA,WAAA;;AACF;AACE,eAAO,qBAAP,IAAO,CAAP;AAVJ;AAFG,GAAA,EAAA,IAAA,CAAP,MAAO,CAAP;AAgBD;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAiFM,SAAA,QAAA,CAAA,IAAA,EAAiD;AACrD,SAAO,IAAA,QAAA,CAAP,IAAO,CAAP;AACD","sourcesContent":["import {\n  ASTv2,\n  generateSyntaxError,\n  isKeyword,\n  KEYWORDS_TYPES,\n  KeywordType,\n} from '@glimmer/syntax';\nimport { exhausted } from '@glimmer/util';\n\nimport { Err, Result } from '../../../shared/result';\nimport { NormalizationState } from '../context';\n\nexport interface KeywordDelegate<Match extends KeywordMatch, V, Out> {\n  assert(options: Match, state: NormalizationState): Result<V>;\n  translate(options: { node: Match; state: NormalizationState }, param: V): Result<Out>;\n}\n\nexport interface Keyword<K extends KeywordType = KeywordType, Out = unknown> {\n  translate(node: KeywordCandidates[K], state: NormalizationState): Result<Out> | null;\n}\n\nexport interface BlockKeyword<Out = unknown> {\n  translate(node: ASTv2.InvokeBlock, state: NormalizationState): Result<Out> | null;\n}\n\nclass KeywordImpl<\n  K extends KeywordType,\n  S extends string = string,\n  Param = unknown,\n  Out = unknown\n> {\n  protected types: Set<KeywordCandidates[K]['type']>;\n\n  constructor(\n    protected keyword: S,\n    type: KeywordType,\n    private delegate: KeywordDelegate<KeywordMatches[K], Param, Out>\n  ) {\n    let nodes = new Set<KeywordNode['type']>();\n    for (let nodeType of KEYWORD_NODES[type]) {\n      nodes.add(nodeType);\n    }\n\n    this.types = nodes;\n  }\n\n  protected match(node: KeywordCandidates[K]): node is KeywordMatches[K] {\n    if (!this.types.has(node.type)) {\n      return false;\n    }\n\n    let path = getPathExpression(node);\n\n    if (path !== null && path.ref.type === 'Free') {\n      if (path.tail.length > 0) {\n        if (path.ref.resolution.serialize() === 'Loose') {\n          // cannot be a keyword reference, keywords do not allow paths (must be\n          // relying on implicit this fallback)\n          return false;\n        }\n      }\n\n      return path.ref.name === this.keyword;\n    } else {\n      return false;\n    }\n  }\n\n  translate(node: KeywordMatches[K], state: NormalizationState): Result<Out> | null {\n    if (this.match(node)) {\n      let path = getPathExpression(node);\n\n      if (path !== null && path.tail.length > 0) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${\n              this.keyword\n            }\\` keyword was used incorrectly. It was used as \\`${path.loc.asString()}\\`, but it cannot be used with additional path segments. \\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n\n      let param = this.delegate.assert(node, state);\n      return param.andThen((param) => this.delegate.translate({ node, state }, param));\n    } else {\n      return null;\n    }\n  }\n}\n\nexport type PossibleNode =\n  | ASTv2.PathExpression\n  | ASTv2.AppendContent\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock;\n\nexport const KEYWORD_NODES = {\n  Call: ['Call'],\n  Block: ['InvokeBlock'],\n  Append: ['AppendContent'],\n  Modifier: ['ElementModifier'],\n} as const;\n\nexport interface KeywordCandidates {\n  Call: ASTv2.ExpressionNode;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordCandidate = KeywordCandidates[keyof KeywordCandidates];\n\nexport interface KeywordMatches {\n  Call: ASTv2.CallExpression;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordMatch = KeywordMatches[keyof KeywordMatches];\n\n/**\n * A \"generic\" keyword is something like `has-block`, which makes sense in the context\n * of sub-expression or append\n */\nexport type GenericKeywordNode = ASTv2.AppendContent | ASTv2.CallExpression;\n\nexport type KeywordNode =\n  | GenericKeywordNode\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock\n  | ASTv2.ElementModifier;\n\nexport function keyword<\n  K extends KeywordType,\n  D extends KeywordDelegate<KeywordMatches[K], unknown, Out>,\n  Out = unknown\n>(keyword: string, type: K, delegate: D): Keyword<K, Out> {\n  return new KeywordImpl(keyword, type, delegate as KeywordDelegate<KeywordMatch, unknown, Out>);\n}\n\nexport type PossibleKeyword = KeywordNode;\ntype OutFor<K extends Keyword | BlockKeyword> = K extends BlockKeyword<infer Out>\n  ? Out\n  : K extends Keyword<KeywordType, infer Out>\n  ? Out\n  : never;\n\nfunction getPathExpression(node: KeywordNode | ASTv2.ExpressionNode): ASTv2.PathExpression | null {\n  switch (node.type) {\n    // This covers the inside of attributes and expressions, as well as the callee\n    // of call nodes\n    case 'Path':\n      return node;\n    case 'AppendContent':\n      return getPathExpression(node.value);\n    case 'Call':\n    case 'InvokeBlock':\n    case 'ElementModifier':\n      return getPathExpression(node.callee);\n    default:\n      return null;\n  }\n}\n\nexport class Keywords<K extends KeywordType, KeywordList extends Keyword<K> = never>\n  implements Keyword<K, OutFor<KeywordList>> {\n  #keywords: Keyword[] = [];\n  #type: K;\n\n  constructor(type: K) {\n    this.#type = type;\n  }\n\n  kw<S extends string = string, Out = unknown>(\n    name: S,\n    delegate: KeywordDelegate<KeywordMatches[K], unknown, Out>\n  ): Keywords<K, KeywordList | Keyword<K, Out>> {\n    this.#keywords.push(keyword(name, this.#type, delegate));\n\n    return this;\n  }\n\n  translate(\n    node: KeywordCandidates[K],\n    state: NormalizationState\n  ): Result<OutFor<KeywordList>> | null {\n    for (let keyword of this.#keywords) {\n      let result = keyword.translate(node, state) as Result<OutFor<KeywordList>>;\n      if (result !== null) {\n        return result;\n      }\n    }\n\n    let path = getPathExpression(node);\n\n    if (path && path.ref.type === 'Free' && isKeyword(path.ref.name)) {\n      let { name } = path.ref;\n\n      let usedType = this.#type;\n      let validTypes = KEYWORDS_TYPES[name];\n\n      if (validTypes.indexOf(usedType) === -1) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${name}\\` keyword was used incorrectly. It was used as ${\n              typesToReadableName[usedType]\n            }, but its valid usages are:\\n\\n${generateTypesMessage(\n              name,\n              validTypes\n            )}\\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n    }\n\n    return null;\n  }\n}\n\nconst typesToReadableName = {\n  Append: 'an append statement',\n  Block: 'a block statement',\n  Call: 'a call expression',\n  Modifier: 'a modifier',\n};\n\nfunction generateTypesMessage(name: string, types: KeywordType[]): string {\n  return types\n    .map((type) => {\n      switch (type) {\n        case 'Append':\n          return `- As an append statement, as in: {{${name}}}`;\n        case 'Block':\n          return `- As a block statement, as in: {{#${name}}}{{/${name}}}`;\n        case 'Call':\n          return `- As an expression, as in: (${name})`;\n        case 'Modifier':\n          return `- As a modifier, as in: <div {{${name}}}></div>`;\n        default:\n          return exhausted(type);\n      }\n    })\n    .join('\\n\\n');\n}\n\n/**\n * This function builds keyword definitions for a particular type of AST node (`KeywordType`).\n *\n * You can build keyword definitions for:\n *\n * - `Expr`: A `SubExpression` or `PathExpression`\n * - `Block`: A `BlockStatement`\n *   - A `BlockStatement` is a keyword candidate if its head is a\n *     `PathExpression`\n * - `Append`: An `AppendStatement`\n *\n * A node is a keyword candidate if:\n *\n * - A `PathExpression` is a keyword candidate if it has no tail, and its\n *   head expression is a `LocalVarHead` or `FreeVarHead` whose name is\n *   the keyword's name.\n * - A `SubExpression`, `AppendStatement`, or `BlockStatement` is a keyword\n *   candidate if its head is a keyword candidate.\n *\n * The keyword infrastructure guarantees that:\n *\n * - If a node is not a keyword candidate, it is never passed to any keyword's\n *   `assert` method.\n * - If a node is not the `KeywordType` for a particular keyword, it will not\n *   be passed to the keyword's `assert` method.\n *\n * `Expr` keywords are used in expression positions and should return HIR\n * expressions. `Block` and `Append` keywords are used in statement\n * positions and should return HIR statements.\n *\n * A keyword definition has two parts:\n *\n * - `match`, which determines whether an AST node matches the keyword, and can\n *   optionally return some information extracted from the AST node.\n * - `translate`, which takes a matching AST node as well as the extracted\n *   information and returns an appropriate HIR instruction.\n *\n * # Example\n *\n * This keyword:\n *\n * - turns `(hello)` into `\"hello\"`\n *   - as long as `hello` is not in scope\n * - makes it an error to pass any arguments (such as `(hello world)`)\n *\n * ```ts\n * keywords('SubExpr').kw('hello', {\n *   assert(node: ExprKeywordNode): Result<void> | false {\n *     // we don't want to transform `hello` as a `PathExpression`\n *     if (node.type !== 'SubExpression') {\n *       return false;\n *     }\n *\n *     // node.head would be `LocalVarHead` if `hello` was in scope\n *     if (node.head.type !== 'FreeVarHead') {\n *       return false;\n *     }\n *\n *     if (node.params.length || node.hash) {\n *       return Err(generateSyntaxError(`(hello) does not take any arguments`), node.loc);\n *     } else {\n *       return Ok();\n *     }\n *   },\n *\n *   translate(node: ASTv2.SubExpression): hir.Expression {\n *     return ASTv2.builders.literal(\"hello\", node.loc)\n *   }\n * })\n * ```\n *\n * The keyword infrastructure checks to make sure that the node is the right\n * type before calling `assert`, so you only need to consider `SubExpression`\n * and `PathExpression` here. It also checks to make sure that the node passed\n * to `assert` has the keyword name in the right place.\n *\n * Note the important difference between returning `false` from `assert`,\n * which just means that the node didn't match, and returning `Err`, which\n * means that the node matched, but there was a keyword-specific syntax\n * error.\n */\nexport function keywords<K extends KeywordType>(type: K): Keywords<K> {\n  return new Keywords(type);\n}\n"],"sourceRoot":""} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../../../packages/@glimmer/compiler/lib/passes/1-normalization/keywords/impl.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ASTv2,\n  generateSyntaxError,\n  isKeyword,\n  KEYWORDS_TYPES,\n  KeywordType,\n} from '@glimmer/syntax';\nimport { exhausted } from '@glimmer/util';\n\nimport { Err, Result } from '../../../shared/result';\nimport { NormalizationState } from '../context';\n\nexport interface KeywordDelegate<Match extends KeywordMatch, V, Out> {\n  assert(options: Match, state: NormalizationState): Result<V>;\n  translate(options: { node: Match; state: NormalizationState }, param: V): Result<Out>;\n}\n\nexport interface Keyword<K extends KeywordType = KeywordType, Out = unknown> {\n  translate(node: KeywordCandidates[K], state: NormalizationState): Result<Out> | null;\n}\n\nexport interface BlockKeyword<Out = unknown> {\n  translate(node: ASTv2.InvokeBlock, state: NormalizationState): Result<Out> | null;\n}\n\nclass KeywordImpl<\n  K extends KeywordType,\n  S extends string = string,\n  Param = unknown,\n  Out = unknown\n> {\n  protected types: Set<KeywordCandidates[K]['type']>;\n\n  constructor(\n    protected keyword: S,\n    type: KeywordType,\n    private delegate: KeywordDelegate<KeywordMatches[K], Param, Out>\n  ) {\n    let nodes = new Set<KeywordNode['type']>();\n    for (let nodeType of KEYWORD_NODES[type]) {\n      nodes.add(nodeType);\n    }\n\n    this.types = nodes;\n  }\n\n  protected match(node: KeywordCandidates[K]): node is KeywordMatches[K] {\n    if (!this.types.has(node.type)) {\n      return false;\n    }\n\n    let path = getCalleeExpression(node);\n\n    if (path !== null && path.type === 'Path' && path.ref.type === 'Free') {\n      if (path.tail.length > 0) {\n        if (path.ref.resolution.serialize() === 'Loose') {\n          // cannot be a keyword reference, keywords do not allow paths (must be\n          // relying on implicit this fallback)\n          return false;\n        }\n      }\n\n      return path.ref.name === this.keyword;\n    } else {\n      return false;\n    }\n  }\n\n  translate(node: KeywordMatches[K], state: NormalizationState): Result<Out> | null {\n    if (this.match(node)) {\n      let path = getCalleeExpression(node);\n\n      if (path !== null && path.type === 'Path' && path.tail.length > 0) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${\n              this.keyword\n            }\\` keyword was used incorrectly. It was used as \\`${path.loc.asString()}\\`, but it cannot be used with additional path segments. \\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n\n      let param = this.delegate.assert(node, state);\n      return param.andThen((param) => this.delegate.translate({ node, state }, param));\n    } else {\n      return null;\n    }\n  }\n}\n\nexport type PossibleNode =\n  | ASTv2.PathExpression\n  | ASTv2.AppendContent\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock;\n\nexport const KEYWORD_NODES = {\n  Call: ['Call'],\n  Block: ['InvokeBlock'],\n  Append: ['AppendContent'],\n  Modifier: ['ElementModifier'],\n} as const;\n\nexport interface KeywordCandidates {\n  Call: ASTv2.ExpressionNode;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordCandidate = KeywordCandidates[keyof KeywordCandidates];\n\nexport interface KeywordMatches {\n  Call: ASTv2.CallExpression;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordMatch = KeywordMatches[keyof KeywordMatches];\n\n/**\n * A \"generic\" keyword is something like `has-block`, which makes sense in the context\n * of sub-expression or append\n */\nexport type GenericKeywordNode = ASTv2.AppendContent | ASTv2.CallExpression;\n\nexport type KeywordNode =\n  | GenericKeywordNode\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock\n  | ASTv2.ElementModifier;\n\nexport function keyword<\n  K extends KeywordType,\n  D extends KeywordDelegate<KeywordMatches[K], unknown, Out>,\n  Out = unknown\n>(keyword: string, type: K, delegate: D): Keyword<K, Out> {\n  return new KeywordImpl(keyword, type, delegate as KeywordDelegate<KeywordMatch, unknown, Out>);\n}\n\nexport type PossibleKeyword = KeywordNode;\ntype OutFor<K extends Keyword | BlockKeyword> = K extends BlockKeyword<infer Out>\n  ? Out\n  : K extends Keyword<KeywordType, infer Out>\n  ? Out\n  : never;\n\nfunction getCalleeExpression(\n  node: KeywordNode | ASTv2.ExpressionNode\n): ASTv2.ExpressionNode | null {\n  switch (node.type) {\n    // This covers the inside of attributes and expressions, as well as the callee\n    // of call nodes\n    case 'Path':\n      return node;\n    case 'AppendContent':\n      return getCalleeExpression(node.value);\n    case 'Call':\n    case 'InvokeBlock':\n    case 'ElementModifier':\n      return node.callee;\n    default:\n      return null;\n  }\n}\n\nexport class Keywords<K extends KeywordType, KeywordList extends Keyword<K> = never>\n  implements Keyword<K, OutFor<KeywordList>> {\n  #keywords: Keyword[] = [];\n  #type: K;\n\n  constructor(type: K) {\n    this.#type = type;\n  }\n\n  kw<S extends string = string, Out = unknown>(\n    name: S,\n    delegate: KeywordDelegate<KeywordMatches[K], unknown, Out>\n  ): Keywords<K, KeywordList | Keyword<K, Out>> {\n    this.#keywords.push(keyword(name, this.#type, delegate));\n\n    return this;\n  }\n\n  translate(\n    node: KeywordCandidates[K],\n    state: NormalizationState\n  ): Result<OutFor<KeywordList>> | null {\n    for (let keyword of this.#keywords) {\n      let result = keyword.translate(node, state) as Result<OutFor<KeywordList>>;\n      if (result !== null) {\n        return result;\n      }\n    }\n\n    let path = getCalleeExpression(node);\n\n    if (path && path.type === 'Path' && path.ref.type === 'Free' && isKeyword(path.ref.name)) {\n      let { name } = path.ref;\n\n      let usedType = this.#type;\n      let validTypes = KEYWORDS_TYPES[name];\n\n      if (validTypes.indexOf(usedType) === -1) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${name}\\` keyword was used incorrectly. It was used as ${\n              typesToReadableName[usedType]\n            }, but its valid usages are:\\n\\n${generateTypesMessage(\n              name,\n              validTypes\n            )}\\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n    }\n\n    return null;\n  }\n}\n\nconst typesToReadableName = {\n  Append: 'an append statement',\n  Block: 'a block statement',\n  Call: 'a call expression',\n  Modifier: 'a modifier',\n};\n\nfunction generateTypesMessage(name: string, types: KeywordType[]): string {\n  return types\n    .map((type) => {\n      switch (type) {\n        case 'Append':\n          return `- As an append statement, as in: {{${name}}}`;\n        case 'Block':\n          return `- As a block statement, as in: {{#${name}}}{{/${name}}}`;\n        case 'Call':\n          return `- As an expression, as in: (${name})`;\n        case 'Modifier':\n          return `- As a modifier, as in: <div {{${name}}}></div>`;\n        default:\n          return exhausted(type);\n      }\n    })\n    .join('\\n\\n');\n}\n\n/**\n * This function builds keyword definitions for a particular type of AST node (`KeywordType`).\n *\n * You can build keyword definitions for:\n *\n * - `Expr`: A `SubExpression` or `PathExpression`\n * - `Block`: A `BlockStatement`\n *   - A `BlockStatement` is a keyword candidate if its head is a\n *     `PathExpression`\n * - `Append`: An `AppendStatement`\n *\n * A node is a keyword candidate if:\n *\n * - A `PathExpression` is a keyword candidate if it has no tail, and its\n *   head expression is a `LocalVarHead` or `FreeVarHead` whose name is\n *   the keyword's name.\n * - A `SubExpression`, `AppendStatement`, or `BlockStatement` is a keyword\n *   candidate if its head is a keyword candidate.\n *\n * The keyword infrastructure guarantees that:\n *\n * - If a node is not a keyword candidate, it is never passed to any keyword's\n *   `assert` method.\n * - If a node is not the `KeywordType` for a particular keyword, it will not\n *   be passed to the keyword's `assert` method.\n *\n * `Expr` keywords are used in expression positions and should return HIR\n * expressions. `Block` and `Append` keywords are used in statement\n * positions and should return HIR statements.\n *\n * A keyword definition has two parts:\n *\n * - `match`, which determines whether an AST node matches the keyword, and can\n *   optionally return some information extracted from the AST node.\n * - `translate`, which takes a matching AST node as well as the extracted\n *   information and returns an appropriate HIR instruction.\n *\n * # Example\n *\n * This keyword:\n *\n * - turns `(hello)` into `\"hello\"`\n *   - as long as `hello` is not in scope\n * - makes it an error to pass any arguments (such as `(hello world)`)\n *\n * ```ts\n * keywords('SubExpr').kw('hello', {\n *   assert(node: ExprKeywordNode): Result<void> | false {\n *     // we don't want to transform `hello` as a `PathExpression`\n *     if (node.type !== 'SubExpression') {\n *       return false;\n *     }\n *\n *     // node.head would be `LocalVarHead` if `hello` was in scope\n *     if (node.head.type !== 'FreeVarHead') {\n *       return false;\n *     }\n *\n *     if (node.params.length || node.hash) {\n *       return Err(generateSyntaxError(`(hello) does not take any arguments`), node.loc);\n *     } else {\n *       return Ok();\n *     }\n *   },\n *\n *   translate(node: ASTv2.SubExpression): hir.Expression {\n *     return ASTv2.builders.literal(\"hello\", node.loc)\n *   }\n * })\n * ```\n *\n * The keyword infrastructure checks to make sure that the node is the right\n * type before calling `assert`, so you only need to consider `SubExpression`\n * and `PathExpression` here. It also checks to make sure that the node passed\n * to `assert` has the keyword name in the right place.\n *\n * Note the important difference between returning `false` from `assert`,\n * which just means that the node didn't match, and returning `Err`, which\n * means that the node matched, but there was a keyword-specific syntax\n * error.\n */\nexport function keywords<K extends KeywordType>(type: K): Keywords<K> {\n  return new Keywords(type);\n}\n"],"sourceRoot":""} |
@@ -1,5 +0,5 @@ | ||
export { precompile, precompileJSON } from './lib/compiler'; | ||
export { defaultId, precompile, precompileJSON } from './lib/compiler'; | ||
export { ProgramSymbols, buildStatement, buildStatements, s, c, unicode, NEWLINE } from './lib/builder/builder'; // exported only for tests | ||
export { default as WireFormatDebugger } from './lib/wire-format-debug'; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3BhY2thZ2VzL0BnbGltbWVyL2NvbXBpbGVyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLFNBQVMsVUFBVCxFQUFxQixjQUFyQixRQUE4RCxnQkFBOUQ7QUFDQSxTQUNFLGNBREYsRUFFRSxjQUZGLEVBR0UsZUFIRixFQUlFLENBSkYsRUFLRSxDQUxGLEVBTUUsT0FORixFQU9FLE9BUEYsUUFRTyx1QkFSUCxDLENBV0E7O0FBQ0EsU0FBUyxPQUFPLElBQUksa0JBQXBCLFFBQThDLHlCQUE5QyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IHByZWNvbXBpbGUsIHByZWNvbXBpbGVKU09OLCBQcmVjb21waWxlT3B0aW9ucyB9IGZyb20gJy4vbGliL2NvbXBpbGVyJztcbmV4cG9ydCB7XG4gIFByb2dyYW1TeW1ib2xzLFxuICBidWlsZFN0YXRlbWVudCxcbiAgYnVpbGRTdGF0ZW1lbnRzLFxuICBzLFxuICBjLFxuICB1bmljb2RlLFxuICBORVdMSU5FLFxufSBmcm9tICcuL2xpYi9idWlsZGVyL2J1aWxkZXInO1xuZXhwb3J0IHsgQnVpbGRlclN0YXRlbWVudCwgQnVpbGRlciB9IGZyb20gJy4vbGliL2J1aWxkZXIvYnVpbGRlci1pbnRlcmZhY2UnO1xuXG4vLyBleHBvcnRlZCBvbmx5IGZvciB0ZXN0c1xuZXhwb3J0IHsgZGVmYXVsdCBhcyBXaXJlRm9ybWF0RGVidWdnZXIgfSBmcm9tICcuL2xpYi93aXJlLWZvcm1hdC1kZWJ1Zyc7XG4iXSwic291cmNlUm9vdCI6IiJ9 | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3BhY2thZ2VzL0BnbGltbWVyL2NvbXBpbGVyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLFNBQVMsU0FBVCxFQUFvQixVQUFwQixFQUFnQyxjQUFoQyxRQUF5RSxnQkFBekU7QUFDQSxTQUNFLGNBREYsRUFFRSxjQUZGLEVBR0UsZUFIRixFQUlFLENBSkYsRUFLRSxDQUxGLEVBTUUsT0FORixFQU9FLE9BUEYsUUFRTyx1QkFSUCxDLENBV0E7O0FBQ0EsU0FBUyxPQUFPLElBQUksa0JBQXBCLFFBQThDLHlCQUE5QyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IGRlZmF1bHRJZCwgcHJlY29tcGlsZSwgcHJlY29tcGlsZUpTT04sIFByZWNvbXBpbGVPcHRpb25zIH0gZnJvbSAnLi9saWIvY29tcGlsZXInO1xuZXhwb3J0IHtcbiAgUHJvZ3JhbVN5bWJvbHMsXG4gIGJ1aWxkU3RhdGVtZW50LFxuICBidWlsZFN0YXRlbWVudHMsXG4gIHMsXG4gIGMsXG4gIHVuaWNvZGUsXG4gIE5FV0xJTkUsXG59IGZyb20gJy4vbGliL2J1aWxkZXIvYnVpbGRlcic7XG5leHBvcnQgeyBCdWlsZGVyU3RhdGVtZW50LCBCdWlsZGVyIH0gZnJvbSAnLi9saWIvYnVpbGRlci9idWlsZGVyLWludGVyZmFjZSc7XG5cbi8vIGV4cG9ydGVkIG9ubHkgZm9yIHRlc3RzXG5leHBvcnQgeyBkZWZhdWx0IGFzIFdpcmVGb3JtYXREZWJ1Z2dlciB9IGZyb20gJy4vbGliL3dpcmUtZm9ybWF0LWRlYnVnJztcbiJdLCJzb3VyY2VSb290IjoiIn0= |
@@ -5,2 +5,28 @@ import { normalize, Source } from '@glimmer/syntax'; | ||
import { visit as pass2 } from './passes/2-encoding/index'; | ||
export const defaultId = (() => { | ||
let req = typeof module === 'object' && typeof module.require === 'function' ? module.require : require; | ||
if (req) { | ||
try { | ||
const crypto = req('crypto'); | ||
let idFn = src => { | ||
let hash = crypto.createHash('sha1'); | ||
hash.update(src, 'utf8'); // trim to 6 bytes of data (2^48 - 1) | ||
return hash.digest('base64').substring(0, 8); | ||
}; | ||
idFn('test'); | ||
return idFn; | ||
} catch (e) {} | ||
} | ||
return function idFn() { | ||
return null; | ||
}; | ||
})(); | ||
const defaultOptions = { | ||
id: defaultId | ||
}; | ||
/* | ||
@@ -21,3 +47,3 @@ * Compile a string into a template javascript string. | ||
export function precompileJSON(string, options = {}) { | ||
export function precompileJSON(string, options = defaultOptions) { | ||
var _a, _b; | ||
@@ -61,3 +87,3 @@ | ||
export function precompile(source, options = {}) { | ||
export function precompile(source, options = defaultOptions) { | ||
var _a, _b; | ||
@@ -67,4 +93,6 @@ | ||
let moduleName = (_a = options.meta) === null || _a === void 0 ? void 0 : _a.moduleName; | ||
let idFn = options.id || defaultId; | ||
let blockJSON = JSON.stringify(block); | ||
let templateJSONObject = { | ||
id: idFn(JSON.stringify(options.meta) + blockJSON), | ||
block: blockJSON, | ||
@@ -94,2 +122,2 @@ moduleName: moduleName !== null && moduleName !== void 0 ? moduleName : '(unknown template module)', | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3BhY2thZ2VzL0BnbGltbWVyL2NvbXBpbGVyL2xpYi9jb21waWxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFNQSxTQUFTLFNBQVQsRUFBdUMsTUFBdkMsUUFBcUQsaUJBQXJEO0FBQ0EsU0FBUyxZQUFULFFBQTZCLGVBQTdCO0FBRUEsT0FBTyxLQUFQLE1BQWtCLGdDQUFsQjtBQUNBLFNBQVMsS0FBSyxJQUFJLEtBQWxCLFFBQStCLDJCQUEvQjtBQUVBOzs7Ozs7Ozs7Ozs7Ozs7QUFjQSxPQUFNLFNBQVUsY0FBVixDQUNKLE1BREksRUFFSixPQUFBLEdBQTZCLEVBRnpCLEVBRTJCOzs7QUFFL0IsTUFBSSxNQUFNLEdBQUcsSUFBSSxNQUFKLENBQVcsTUFBWCxFQUFpQixDQUFBLEVBQUEsR0FBRSxPQUFPLENBQUMsSUFBVixNQUFjLElBQWQsSUFBYyxFQUFBLEtBQUEsS0FBQSxDQUFkLEdBQWMsS0FBQSxDQUFkLEdBQWMsRUFBQSxDQUFFLFVBQWpDLENBQWI7QUFDQSxNQUFJLENBQUMsR0FBRCxFQUFNLE1BQU4sSUFBZ0IsU0FBUyxDQUFDLE1BQUQsRUFBUyxPQUFULENBQTdCO0FBQ0EsTUFBSSxLQUFLLEdBQUcsS0FBSyxDQUFDLE1BQUQsRUFBUyxHQUFULEVBQVksQ0FBQSxFQUFBLEdBQUUsT0FBTyxDQUFDLFVBQVYsTUFBb0IsSUFBcEIsSUFBb0IsRUFBQSxLQUFBLEtBQUEsQ0FBcEIsR0FBb0IsRUFBcEIsR0FBd0IsS0FBcEMsQ0FBTCxDQUFnRCxLQUFoRCxDQUF1RCxPQUFELElBQVk7QUFDNUUsV0FBTyxLQUFLLENBQUMsT0FBRCxDQUFaO0FBQ0QsR0FGVyxDQUFaOztBQUlBO0FBQUE7QUFBQSxJQUFzQjtBQUNwQixJQUFBLFlBQVksQ0FBQyxHQUFiLENBQWlCLGFBQWpCLEVBQWdDLEtBQWhDO0FBQ0Q7O0FBRUQsTUFBSSxLQUFLLENBQUMsSUFBVixFQUFnQjtBQUNkLFdBQU8sQ0FBQyxLQUFLLENBQUMsS0FBUCxFQUFjLE1BQWQsQ0FBUDtBQUNELEdBRkQsTUFFTztBQUNMLFVBQU0sS0FBSyxDQUFDLE1BQVo7QUFDRDtBQUNGLEMsQ0FFRDtBQUNBOztBQUNBLE1BQU0saUJBQWlCLEdBQUcsc0NBQTFCO0FBRUE7Ozs7Ozs7Ozs7Ozs7OztBQWNBLE9BQU0sU0FBVSxVQUFWLENBQXFCLE1BQXJCLEVBQXFDLE9BQUEsR0FBNkIsRUFBbEUsRUFBb0U7OztBQUN4RSxNQUFJLENBQUMsS0FBRCxFQUFRLFVBQVIsSUFBc0IsY0FBYyxDQUFDLE1BQUQsRUFBUyxPQUFULENBQXhDO0FBRUEsTUFBSSxVQUFVLEdBQUEsQ0FBQSxFQUFBLEdBQUcsT0FBTyxDQUFDLElBQVgsTUFBZSxJQUFmLElBQWUsRUFBQSxLQUFBLEtBQUEsQ0FBZixHQUFlLEtBQUEsQ0FBZixHQUFlLEVBQUEsQ0FBRSxVQUEvQjtBQUNBLE1BQUksU0FBUyxHQUFHLElBQUksQ0FBQyxTQUFMLENBQWUsS0FBZixDQUFoQjtBQUNBLE1BQUksa0JBQWtCLEdBQW9DO0FBQ3hELElBQUEsS0FBSyxFQUFFLFNBRGlEO0FBRXhELElBQUEsVUFBVSxFQUFFLFVBQVUsS0FBQSxJQUFWLElBQUEsVUFBVSxLQUFBLEtBQUEsQ0FBVixHQUFBLFVBQUEsR0FBYywyQkFGOEI7QUFHeEQ7QUFDQTtBQUNBLElBQUEsS0FBSyxFQUFHLGlCQUxnRDtBQU14RCxJQUFBLFlBQVksRUFBQSxDQUFBLEVBQUEsR0FBRSxPQUFPLENBQUMsVUFBVixNQUFvQixJQUFwQixJQUFvQixFQUFBLEtBQUEsS0FBQSxDQUFwQixHQUFvQixFQUFwQixHQUF3QjtBQU5vQixHQUExRDs7QUFTQSxNQUFJLENBQUMsT0FBTyxDQUFDLFVBQWIsRUFBeUI7QUFDdkIsV0FBTyxrQkFBa0IsQ0FBQyxLQUExQjtBQUNELEdBaEJ1RSxDQWtCeEU7OztBQUNBLE1BQUksV0FBVyxHQUFHLElBQUksQ0FBQyxTQUFMLENBQWUsa0JBQWYsQ0FBbEI7O0FBRUEsTUFBSSxPQUFPLENBQUMsVUFBUixJQUFzQixVQUFVLENBQUMsTUFBWCxHQUFvQixDQUE5QyxFQUFpRDtBQUMvQyxRQUFJLE9BQU8sR0FBRyxRQUFRLFVBQVUsQ0FBQyxJQUFYLENBQWdCLEdBQWhCLENBQW9CLEdBQTFDO0FBRUEsSUFBQSxXQUFXLEdBQUcsV0FBVyxDQUFDLE9BQVosQ0FBb0IsSUFBSSxpQkFBaUIsR0FBekMsRUFBOEMsT0FBOUMsQ0FBZDtBQUNELEdBSkQsTUFJTztBQUNMLElBQUEsV0FBVyxHQUFHLFdBQVcsQ0FBQyxPQUFaLENBQW9CLElBQUksaUJBQWlCLEdBQXpDLEVBQThDLE1BQTlDLENBQWQ7QUFDRDs7QUFFRCxTQUFPLFdBQVA7QUFDRCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIFNlcmlhbGl6ZWRUZW1wbGF0ZUJsb2NrLFxuICBTZXJpYWxpemVkVGVtcGxhdGVXaXRoTGF6eUJsb2NrLFxuICBUZW1wbGF0ZUphdmFzY3JpcHQsXG59IGZyb20gJ0BnbGltbWVyL2ludGVyZmFjZXMnO1xuaW1wb3J0IHsgTE9DQUxfU0hPVUxEX0xPRyB9IGZyb20gJ0BnbGltbWVyL2xvY2FsLWRlYnVnLWZsYWdzJztcbmltcG9ydCB7IG5vcm1hbGl6ZSwgUHJlY29tcGlsZU9wdGlvbnMsIFNvdXJjZSB9IGZyb20gJ0BnbGltbWVyL3N5bnRheCc7XG5pbXBvcnQgeyBMT0NBTF9MT0dHRVIgfSBmcm9tICdAZ2xpbW1lci91dGlsJztcblxuaW1wb3J0IHBhc3MwIGZyb20gJy4vcGFzc2VzLzEtbm9ybWFsaXphdGlvbi9pbmRleCc7XG5pbXBvcnQgeyB2aXNpdCBhcyBwYXNzMiB9IGZyb20gJy4vcGFzc2VzLzItZW5jb2RpbmcvaW5kZXgnO1xuXG4vKlxuICogQ29tcGlsZSBhIHN0cmluZyBpbnRvIGEgdGVtcGxhdGUgamF2YXNjcmlwdCBzdHJpbmcuXG4gKlxuICogRXhhbXBsZSB1c2FnZTpcbiAqICAgICBpbXBvcnQgeyBwcmVjb21waWxlIH0gZnJvbSAnQGdsaW1tZXIvY29tcGlsZXInO1xuICogICAgIGltcG9ydCB7IHRlbXBsYXRlRmFjdG9yeSB9IGZyb20gJ2dsaW1tZXItcnVudGltZSc7XG4gKiAgICAgbGV0IHRlbXBsYXRlSnMgPSBwcmVjb21waWxlKFwiSG93ZHkge3tuYW1lfX1cIik7XG4gKiAgICAgbGV0IGZhY3RvcnkgPSB0ZW1wbGF0ZUZhY3RvcnkobmV3IEZ1bmN0aW9uKFwicmV0dXJuIFwiICsgdGVtcGxhdGVKcykoKSk7XG4gKiAgICAgbGV0IHRlbXBsYXRlID0gZmFjdG9yeS5jcmVhdGUoZW52KTtcbiAqXG4gKiBAbWV0aG9kIHByZWNvbXBpbGVcbiAqIEBwYXJhbSB7c3RyaW5nfSBzdHJpbmcgYSBHbGltbWVyIHRlbXBsYXRlIHN0cmluZ1xuICogQHJldHVybiB7c3RyaW5nfSBhIHRlbXBsYXRlIGphdmFzY3JpcHQgc3RyaW5nXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBwcmVjb21waWxlSlNPTihcbiAgc3RyaW5nOiBzdHJpbmcsXG4gIG9wdGlvbnM6IFByZWNvbXBpbGVPcHRpb25zID0ge31cbik6IFtibG9jazogU2VyaWFsaXplZFRlbXBsYXRlQmxvY2ssIHVzZWRMb2NhbHM6IHN0cmluZ1tdXSB7XG4gIGxldCBzb3VyY2UgPSBuZXcgU291cmNlKHN0cmluZywgb3B0aW9ucy5tZXRhPy5tb2R1bGVOYW1lKTtcbiAgbGV0IFthc3QsIGxvY2Fsc10gPSBub3JtYWxpemUoc291cmNlLCBvcHRpb25zKTtcbiAgbGV0IGJsb2NrID0gcGFzczAoc291cmNlLCBhc3QsIG9wdGlvbnMuc3RyaWN0TW9kZSA/PyBmYWxzZSkubWFwT2soKHBhc3MySW4pID0+IHtcbiAgICByZXR1cm4gcGFzczIocGFzczJJbik7XG4gIH0pO1xuXG4gIGlmIChMT0NBTF9TSE9VTERfTE9HKSB7XG4gICAgTE9DQUxfTE9HR0VSLmxvZyhgVGVtcGxhdGUgLT5gLCBibG9jayk7XG4gIH1cblxuICBpZiAoYmxvY2suaXNPaykge1xuICAgIHJldHVybiBbYmxvY2sudmFsdWUsIGxvY2Fsc107XG4gIH0gZWxzZSB7XG4gICAgdGhyb3cgYmxvY2sucmVhc29uO1xuICB9XG59XG5cbi8vIFVVSUQgdXNlZCBhcyBhIHVuaXF1ZSBwbGFjZWhvbGRlciBmb3IgcGxhY2luZyBhIHNuaXBwZXQgb2YgSlMgY29kZSBpbnRvXG4vLyB0aGUgb3RoZXJ3aXNlIEpTT04gc3RyaW5naWZpZWQgdmFsdWUgYmVsb3cuXG5jb25zdCBTQ09QRV9QTEFDRUhPTERFUiA9ICc3OTZkMjRlNi0yNDUwLTRmYjAtOGNkZi1iNjU2MzhiNWVmNzAnO1xuXG4vKlxuICogQ29tcGlsZSBhIHN0cmluZyBpbnRvIGEgdGVtcGxhdGUgamF2YXNjcmlwdCBzdHJpbmcuXG4gKlxuICogRXhhbXBsZSB1c2FnZTpcbiAqICAgICBpbXBvcnQgeyBwcmVjb21waWxlIH0gZnJvbSAnQGdsaW1tZXIvY29tcGlsZXInO1xuICogICAgIGltcG9ydCB7IHRlbXBsYXRlRmFjdG9yeSB9IGZyb20gJ2dsaW1tZXItcnVudGltZSc7XG4gKiAgICAgbGV0IHRlbXBsYXRlSnMgPSBwcmVjb21waWxlKFwiSG93ZHkge3tuYW1lfX1cIik7XG4gKiAgICAgbGV0IGZhY3RvcnkgPSB0ZW1wbGF0ZUZhY3RvcnkobmV3IEZ1bmN0aW9uKFwicmV0dXJuIFwiICsgdGVtcGxhdGVKcykoKSk7XG4gKiAgICAgbGV0IHRlbXBsYXRlID0gZmFjdG9yeS5jcmVhdGUoZW52KTtcbiAqXG4gKiBAbWV0aG9kIHByZWNvbXBpbGVcbiAqIEBwYXJhbSB7c3RyaW5nfSBzdHJpbmcgYSBHbGltbWVyIHRlbXBsYXRlIHN0cmluZ1xuICogQHJldHVybiB7c3RyaW5nfSBhIHRlbXBsYXRlIGphdmFzY3JpcHQgc3RyaW5nXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBwcmVjb21waWxlKHNvdXJjZTogc3RyaW5nLCBvcHRpb25zOiBQcmVjb21waWxlT3B0aW9ucyA9IHt9KTogVGVtcGxhdGVKYXZhc2NyaXB0IHtcbiAgbGV0IFtibG9jaywgdXNlZExvY2Fsc10gPSBwcmVjb21waWxlSlNPTihzb3VyY2UsIG9wdGlvbnMpO1xuXG4gIGxldCBtb2R1bGVOYW1lID0gb3B0aW9ucy5tZXRhPy5tb2R1bGVOYW1lO1xuICBsZXQgYmxvY2tKU09OID0gSlNPTi5zdHJpbmdpZnkoYmxvY2spO1xuICBsZXQgdGVtcGxhdGVKU09OT2JqZWN0OiBTZXJpYWxpemVkVGVtcGxhdGVXaXRoTGF6eUJsb2NrID0ge1xuICAgIGJsb2NrOiBibG9ja0pTT04sXG4gICAgbW9kdWxlTmFtZTogbW9kdWxlTmFtZSA/PyAnKHVua25vd24gdGVtcGxhdGUgbW9kdWxlKScsXG4gICAgLy8gbHlpbmcgdG8gdGhlIHR5cGUgY2hlY2tlciBoZXJlIGJlY2F1c2Ugd2UncmUgZ29pbmcgdG9cbiAgICAvLyByZXBsYWNlIGl0IGp1c3QgYmVsb3csIGFmdGVyIHN0cmluZ2lmaWNhdGlvblxuICAgIHNjb3BlOiAoU0NPUEVfUExBQ0VIT0xERVIgYXMgdW5rbm93bikgYXMgbnVsbCxcbiAgICBpc1N0cmljdE1vZGU6IG9wdGlvbnMuc3RyaWN0TW9kZSA/PyBmYWxzZSxcbiAgfTtcblxuICBpZiAoIW9wdGlvbnMuc3RyaWN0TW9kZSkge1xuICAgIGRlbGV0ZSB0ZW1wbGF0ZUpTT05PYmplY3Quc2NvcGU7XG4gIH1cblxuICAvLyBKU09OIGlzIGphdmFzY3JpcHRcbiAgbGV0IHN0cmluZ2lmaWVkID0gSlNPTi5zdHJpbmdpZnkodGVtcGxhdGVKU09OT2JqZWN0KTtcblxuICBpZiAob3B0aW9ucy5zdHJpY3RNb2RlICYmIHVzZWRMb2NhbHMubGVuZ3RoID4gMCkge1xuICAgIGxldCBzY29wZUZuID0gYCgpPT5bJHt1c2VkTG9jYWxzLmpvaW4oJywnKX1dYDtcblxuICAgIHN0cmluZ2lmaWVkID0gc3RyaW5naWZpZWQucmVwbGFjZShgXCIke1NDT1BFX1BMQUNFSE9MREVSfVwiYCwgc2NvcGVGbik7XG4gIH0gZWxzZSB7XG4gICAgc3RyaW5naWZpZWQgPSBzdHJpbmdpZmllZC5yZXBsYWNlKGBcIiR7U0NPUEVfUExBQ0VIT0xERVJ9XCJgLCAnbnVsbCcpO1xuICB9XG5cbiAgcmV0dXJuIHN0cmluZ2lmaWVkO1xufVxuXG5leHBvcnQgeyBQcmVjb21waWxlT3B0aW9ucyB9O1xuIl0sInNvdXJjZVJvb3QiOiIifQ== | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../packages/@glimmer/compiler/lib/compiler.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  SerializedTemplateBlock,\n  SerializedTemplateWithLazyBlock,\n  TemplateJavascript,\n} from '@glimmer/interfaces';\nimport { LOCAL_SHOULD_LOG } from '@glimmer/local-debug-flags';\nimport { normalize, PrecompileOptions, Source, TemplateIdFn } from '@glimmer/syntax';\nimport { LOCAL_LOGGER } from '@glimmer/util';\n\nimport pass0 from './passes/1-normalization/index';\nimport { visit as pass2 } from './passes/2-encoding/index';\n\ndeclare function require(id: 'crypto'): Crypto;\ndeclare function require(id: string): unknown;\n\ninterface Crypto {\n  createHash(\n    alg: 'sha1'\n  ): {\n    update(src: string, encoding: 'utf8'): void;\n    digest(encoding: 'base64'): string;\n  };\n}\n\nexport const defaultId: TemplateIdFn = (() => {\n  let req: typeof require | undefined =\n    typeof module === 'object' && typeof module.require === 'function' ? module.require : require;\n\n  if (req) {\n    try {\n      const crypto = req('crypto');\n\n      let idFn: TemplateIdFn = (src) => {\n        let hash = crypto.createHash('sha1');\n        hash.update(src, 'utf8');\n        // trim to 6 bytes of data (2^48 - 1)\n        return hash.digest('base64').substring(0, 8);\n      };\n\n      idFn('test');\n\n      return idFn;\n    } catch (e) {}\n  }\n\n  return function idFn() {\n    return null;\n  };\n})();\n\nconst defaultOptions: PrecompileOptions = {\n  id: defaultId,\n};\n\n/*\n * Compile a string into a template javascript string.\n *\n * Example usage:\n *     import { precompile } from '@glimmer/compiler';\n *     import { templateFactory } from 'glimmer-runtime';\n *     let templateJs = precompile(\"Howdy {{name}}\");\n *     let factory = templateFactory(new Function(\"return \" + templateJs)());\n *     let template = factory.create(env);\n *\n * @method precompile\n * @param {string} string a Glimmer template string\n * @return {string} a template javascript string\n */\nexport function precompileJSON(\n  string: string,\n  options: PrecompileOptions = defaultOptions\n): [block: SerializedTemplateBlock, usedLocals: string[]] {\n  let source = new Source(string, options.meta?.moduleName);\n  let [ast, locals] = normalize(source, options);\n  let block = pass0(source, ast, options.strictMode ?? false).mapOk((pass2In) => {\n    return pass2(pass2In);\n  });\n\n  if (LOCAL_SHOULD_LOG) {\n    LOCAL_LOGGER.log(`Template ->`, block);\n  }\n\n  if (block.isOk) {\n    return [block.value, locals];\n  } else {\n    throw block.reason;\n  }\n}\n\n// UUID used as a unique placeholder for placing a snippet of JS code into\n// the otherwise JSON stringified value below.\nconst SCOPE_PLACEHOLDER = '796d24e6-2450-4fb0-8cdf-b65638b5ef70';\n\n/*\n * Compile a string into a template javascript string.\n *\n * Example usage:\n *     import { precompile } from '@glimmer/compiler';\n *     import { templateFactory } from 'glimmer-runtime';\n *     let templateJs = precompile(\"Howdy {{name}}\");\n *     let factory = templateFactory(new Function(\"return \" + templateJs)());\n *     let template = factory.create(env);\n *\n * @method precompile\n * @param {string} string a Glimmer template string\n * @return {string} a template javascript string\n */\nexport function precompile(\n  source: string,\n  options: PrecompileOptions = defaultOptions\n): TemplateJavascript {\n  let [block, usedLocals] = precompileJSON(source, options);\n\n  let moduleName = options.meta?.moduleName;\n  let idFn = options.id || defaultId;\n  let blockJSON = JSON.stringify(block);\n  let templateJSONObject: SerializedTemplateWithLazyBlock = {\n    id: idFn(JSON.stringify(options.meta) + blockJSON),\n    block: blockJSON,\n    moduleName: moduleName ?? '(unknown template module)',\n    // lying to the type checker here because we're going to\n    // replace it just below, after stringification\n    scope: (SCOPE_PLACEHOLDER as unknown) as null,\n    isStrictMode: options.strictMode ?? false,\n  };\n\n  if (!options.strictMode) {\n    delete templateJSONObject.scope;\n  }\n\n  // JSON is javascript\n  let stringified = JSON.stringify(templateJSONObject);\n\n  if (options.strictMode && usedLocals.length > 0) {\n    let scopeFn = `()=>[${usedLocals.join(',')}]`;\n\n    stringified = stringified.replace(`\"${SCOPE_PLACEHOLDER}\"`, scopeFn);\n  } else {\n    stringified = stringified.replace(`\"${SCOPE_PLACEHOLDER}\"`, 'null');\n  }\n\n  return stringified;\n}\n\nexport { PrecompileOptions };\n"],"sourceRoot":""} |
@@ -42,5 +42,5 @@ var __classPrivateFieldSet = this && this.__classPrivateFieldSet || function (receiver, privateMap, value) { | ||
let path = getPathExpression(node); | ||
let path = getCalleeExpression(node); | ||
if (path !== null && path.ref.type === 'Free') { | ||
if (path !== null && path.type === 'Path' && path.ref.type === 'Free') { | ||
if (path.tail.length > 0) { | ||
@@ -62,5 +62,5 @@ if (path.ref.resolution.serialize() === 'Loose') { | ||
if (this.match(node)) { | ||
let path = getPathExpression(node); | ||
let path = getCalleeExpression(node); | ||
if (path !== null && path.tail.length > 0) { | ||
if (path !== null && path.type === 'Path' && path.tail.length > 0) { | ||
return Err(generateSyntaxError(`The \`${this.keyword}\` keyword was used incorrectly. It was used as \`${path.loc.asString()}\`, but it cannot be used with additional path segments. \n\nError caused by`, node.loc)); | ||
@@ -91,3 +91,3 @@ } | ||
function getPathExpression(node) { | ||
function getCalleeExpression(node) { | ||
switch (node.type) { | ||
@@ -100,3 +100,3 @@ // This covers the inside of attributes and expressions, as well as the callee | ||
case 'AppendContent': | ||
return getPathExpression(node.value); | ||
return getCalleeExpression(node.value); | ||
@@ -106,3 +106,3 @@ case 'Call': | ||
case 'ElementModifier': | ||
return getPathExpression(node.callee); | ||
return node.callee; | ||
@@ -138,5 +138,5 @@ default: | ||
let path = getPathExpression(node); | ||
let path = getCalleeExpression(node); | ||
if (path && path.ref.type === 'Free' && isKeyword(path.ref.name)) { | ||
if (path && path.type === 'Path' && path.ref.type === 'Free' && isKeyword(path.ref.name)) { | ||
let { | ||
@@ -273,2 +273,2 @@ name | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../../../packages/@glimmer/compiler/lib/passes/1-normalization/keywords/impl.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ASTv2,\n  generateSyntaxError,\n  isKeyword,\n  KEYWORDS_TYPES,\n  KeywordType,\n} from '@glimmer/syntax';\nimport { exhausted } from '@glimmer/util';\n\nimport { Err, Result } from '../../../shared/result';\nimport { NormalizationState } from '../context';\n\nexport interface KeywordDelegate<Match extends KeywordMatch, V, Out> {\n  assert(options: Match, state: NormalizationState): Result<V>;\n  translate(options: { node: Match; state: NormalizationState }, param: V): Result<Out>;\n}\n\nexport interface Keyword<K extends KeywordType = KeywordType, Out = unknown> {\n  translate(node: KeywordCandidates[K], state: NormalizationState): Result<Out> | null;\n}\n\nexport interface BlockKeyword<Out = unknown> {\n  translate(node: ASTv2.InvokeBlock, state: NormalizationState): Result<Out> | null;\n}\n\nclass KeywordImpl<\n  K extends KeywordType,\n  S extends string = string,\n  Param = unknown,\n  Out = unknown\n> {\n  protected types: Set<KeywordCandidates[K]['type']>;\n\n  constructor(\n    protected keyword: S,\n    type: KeywordType,\n    private delegate: KeywordDelegate<KeywordMatches[K], Param, Out>\n  ) {\n    let nodes = new Set<KeywordNode['type']>();\n    for (let nodeType of KEYWORD_NODES[type]) {\n      nodes.add(nodeType);\n    }\n\n    this.types = nodes;\n  }\n\n  protected match(node: KeywordCandidates[K]): node is KeywordMatches[K] {\n    if (!this.types.has(node.type)) {\n      return false;\n    }\n\n    let path = getPathExpression(node);\n\n    if (path !== null && path.ref.type === 'Free') {\n      if (path.tail.length > 0) {\n        if (path.ref.resolution.serialize() === 'Loose') {\n          // cannot be a keyword reference, keywords do not allow paths (must be\n          // relying on implicit this fallback)\n          return false;\n        }\n      }\n\n      return path.ref.name === this.keyword;\n    } else {\n      return false;\n    }\n  }\n\n  translate(node: KeywordMatches[K], state: NormalizationState): Result<Out> | null {\n    if (this.match(node)) {\n      let path = getPathExpression(node);\n\n      if (path !== null && path.tail.length > 0) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${\n              this.keyword\n            }\\` keyword was used incorrectly. It was used as \\`${path.loc.asString()}\\`, but it cannot be used with additional path segments. \\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n\n      let param = this.delegate.assert(node, state);\n      return param.andThen((param) => this.delegate.translate({ node, state }, param));\n    } else {\n      return null;\n    }\n  }\n}\n\nexport type PossibleNode =\n  | ASTv2.PathExpression\n  | ASTv2.AppendContent\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock;\n\nexport const KEYWORD_NODES = {\n  Call: ['Call'],\n  Block: ['InvokeBlock'],\n  Append: ['AppendContent'],\n  Modifier: ['ElementModifier'],\n} as const;\n\nexport interface KeywordCandidates {\n  Call: ASTv2.ExpressionNode;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordCandidate = KeywordCandidates[keyof KeywordCandidates];\n\nexport interface KeywordMatches {\n  Call: ASTv2.CallExpression;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordMatch = KeywordMatches[keyof KeywordMatches];\n\n/**\n * A \"generic\" keyword is something like `has-block`, which makes sense in the context\n * of sub-expression or append\n */\nexport type GenericKeywordNode = ASTv2.AppendContent | ASTv2.CallExpression;\n\nexport type KeywordNode =\n  | GenericKeywordNode\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock\n  | ASTv2.ElementModifier;\n\nexport function keyword<\n  K extends KeywordType,\n  D extends KeywordDelegate<KeywordMatches[K], unknown, Out>,\n  Out = unknown\n>(keyword: string, type: K, delegate: D): Keyword<K, Out> {\n  return new KeywordImpl(keyword, type, delegate as KeywordDelegate<KeywordMatch, unknown, Out>);\n}\n\nexport type PossibleKeyword = KeywordNode;\ntype OutFor<K extends Keyword | BlockKeyword> = K extends BlockKeyword<infer Out>\n  ? Out\n  : K extends Keyword<KeywordType, infer Out>\n  ? Out\n  : never;\n\nfunction getPathExpression(node: KeywordNode | ASTv2.ExpressionNode): ASTv2.PathExpression | null {\n  switch (node.type) {\n    // This covers the inside of attributes and expressions, as well as the callee\n    // of call nodes\n    case 'Path':\n      return node;\n    case 'AppendContent':\n      return getPathExpression(node.value);\n    case 'Call':\n    case 'InvokeBlock':\n    case 'ElementModifier':\n      return getPathExpression(node.callee);\n    default:\n      return null;\n  }\n}\n\nexport class Keywords<K extends KeywordType, KeywordList extends Keyword<K> = never>\n  implements Keyword<K, OutFor<KeywordList>> {\n  #keywords: Keyword[] = [];\n  #type: K;\n\n  constructor(type: K) {\n    this.#type = type;\n  }\n\n  kw<S extends string = string, Out = unknown>(\n    name: S,\n    delegate: KeywordDelegate<KeywordMatches[K], unknown, Out>\n  ): Keywords<K, KeywordList | Keyword<K, Out>> {\n    this.#keywords.push(keyword(name, this.#type, delegate));\n\n    return this;\n  }\n\n  translate(\n    node: KeywordCandidates[K],\n    state: NormalizationState\n  ): Result<OutFor<KeywordList>> | null {\n    for (let keyword of this.#keywords) {\n      let result = keyword.translate(node, state) as Result<OutFor<KeywordList>>;\n      if (result !== null) {\n        return result;\n      }\n    }\n\n    let path = getPathExpression(node);\n\n    if (path && path.ref.type === 'Free' && isKeyword(path.ref.name)) {\n      let { name } = path.ref;\n\n      let usedType = this.#type;\n      let validTypes = KEYWORDS_TYPES[name];\n\n      if (validTypes.indexOf(usedType) === -1) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${name}\\` keyword was used incorrectly. It was used as ${\n              typesToReadableName[usedType]\n            }, but its valid usages are:\\n\\n${generateTypesMessage(\n              name,\n              validTypes\n            )}\\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n    }\n\n    return null;\n  }\n}\n\nconst typesToReadableName = {\n  Append: 'an append statement',\n  Block: 'a block statement',\n  Call: 'a call expression',\n  Modifier: 'a modifier',\n};\n\nfunction generateTypesMessage(name: string, types: KeywordType[]): string {\n  return types\n    .map((type) => {\n      switch (type) {\n        case 'Append':\n          return `- As an append statement, as in: {{${name}}}`;\n        case 'Block':\n          return `- As a block statement, as in: {{#${name}}}{{/${name}}}`;\n        case 'Call':\n          return `- As an expression, as in: (${name})`;\n        case 'Modifier':\n          return `- As a modifier, as in: <div {{${name}}}></div>`;\n        default:\n          return exhausted(type);\n      }\n    })\n    .join('\\n\\n');\n}\n\n/**\n * This function builds keyword definitions for a particular type of AST node (`KeywordType`).\n *\n * You can build keyword definitions for:\n *\n * - `Expr`: A `SubExpression` or `PathExpression`\n * - `Block`: A `BlockStatement`\n *   - A `BlockStatement` is a keyword candidate if its head is a\n *     `PathExpression`\n * - `Append`: An `AppendStatement`\n *\n * A node is a keyword candidate if:\n *\n * - A `PathExpression` is a keyword candidate if it has no tail, and its\n *   head expression is a `LocalVarHead` or `FreeVarHead` whose name is\n *   the keyword's name.\n * - A `SubExpression`, `AppendStatement`, or `BlockStatement` is a keyword\n *   candidate if its head is a keyword candidate.\n *\n * The keyword infrastructure guarantees that:\n *\n * - If a node is not a keyword candidate, it is never passed to any keyword's\n *   `assert` method.\n * - If a node is not the `KeywordType` for a particular keyword, it will not\n *   be passed to the keyword's `assert` method.\n *\n * `Expr` keywords are used in expression positions and should return HIR\n * expressions. `Block` and `Append` keywords are used in statement\n * positions and should return HIR statements.\n *\n * A keyword definition has two parts:\n *\n * - `match`, which determines whether an AST node matches the keyword, and can\n *   optionally return some information extracted from the AST node.\n * - `translate`, which takes a matching AST node as well as the extracted\n *   information and returns an appropriate HIR instruction.\n *\n * # Example\n *\n * This keyword:\n *\n * - turns `(hello)` into `\"hello\"`\n *   - as long as `hello` is not in scope\n * - makes it an error to pass any arguments (such as `(hello world)`)\n *\n * ```ts\n * keywords('SubExpr').kw('hello', {\n *   assert(node: ExprKeywordNode): Result<void> | false {\n *     // we don't want to transform `hello` as a `PathExpression`\n *     if (node.type !== 'SubExpression') {\n *       return false;\n *     }\n *\n *     // node.head would be `LocalVarHead` if `hello` was in scope\n *     if (node.head.type !== 'FreeVarHead') {\n *       return false;\n *     }\n *\n *     if (node.params.length || node.hash) {\n *       return Err(generateSyntaxError(`(hello) does not take any arguments`), node.loc);\n *     } else {\n *       return Ok();\n *     }\n *   },\n *\n *   translate(node: ASTv2.SubExpression): hir.Expression {\n *     return ASTv2.builders.literal(\"hello\", node.loc)\n *   }\n * })\n * ```\n *\n * The keyword infrastructure checks to make sure that the node is the right\n * type before calling `assert`, so you only need to consider `SubExpression`\n * and `PathExpression` here. It also checks to make sure that the node passed\n * to `assert` has the keyword name in the right place.\n *\n * Note the important difference between returning `false` from `assert`,\n * which just means that the node didn't match, and returning `Err`, which\n * means that the node matched, but there was a keyword-specific syntax\n * error.\n */\nexport function keywords<K extends KeywordType>(type: K): Keywords<K> {\n  return new Keywords(type);\n}\n"],"sourceRoot":""} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../../../packages/@glimmer/compiler/lib/passes/1-normalization/keywords/impl.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ASTv2,\n  generateSyntaxError,\n  isKeyword,\n  KEYWORDS_TYPES,\n  KeywordType,\n} from '@glimmer/syntax';\nimport { exhausted } from '@glimmer/util';\n\nimport { Err, Result } from '../../../shared/result';\nimport { NormalizationState } from '../context';\n\nexport interface KeywordDelegate<Match extends KeywordMatch, V, Out> {\n  assert(options: Match, state: NormalizationState): Result<V>;\n  translate(options: { node: Match; state: NormalizationState }, param: V): Result<Out>;\n}\n\nexport interface Keyword<K extends KeywordType = KeywordType, Out = unknown> {\n  translate(node: KeywordCandidates[K], state: NormalizationState): Result<Out> | null;\n}\n\nexport interface BlockKeyword<Out = unknown> {\n  translate(node: ASTv2.InvokeBlock, state: NormalizationState): Result<Out> | null;\n}\n\nclass KeywordImpl<\n  K extends KeywordType,\n  S extends string = string,\n  Param = unknown,\n  Out = unknown\n> {\n  protected types: Set<KeywordCandidates[K]['type']>;\n\n  constructor(\n    protected keyword: S,\n    type: KeywordType,\n    private delegate: KeywordDelegate<KeywordMatches[K], Param, Out>\n  ) {\n    let nodes = new Set<KeywordNode['type']>();\n    for (let nodeType of KEYWORD_NODES[type]) {\n      nodes.add(nodeType);\n    }\n\n    this.types = nodes;\n  }\n\n  protected match(node: KeywordCandidates[K]): node is KeywordMatches[K] {\n    if (!this.types.has(node.type)) {\n      return false;\n    }\n\n    let path = getCalleeExpression(node);\n\n    if (path !== null && path.type === 'Path' && path.ref.type === 'Free') {\n      if (path.tail.length > 0) {\n        if (path.ref.resolution.serialize() === 'Loose') {\n          // cannot be a keyword reference, keywords do not allow paths (must be\n          // relying on implicit this fallback)\n          return false;\n        }\n      }\n\n      return path.ref.name === this.keyword;\n    } else {\n      return false;\n    }\n  }\n\n  translate(node: KeywordMatches[K], state: NormalizationState): Result<Out> | null {\n    if (this.match(node)) {\n      let path = getCalleeExpression(node);\n\n      if (path !== null && path.type === 'Path' && path.tail.length > 0) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${\n              this.keyword\n            }\\` keyword was used incorrectly. It was used as \\`${path.loc.asString()}\\`, but it cannot be used with additional path segments. \\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n\n      let param = this.delegate.assert(node, state);\n      return param.andThen((param) => this.delegate.translate({ node, state }, param));\n    } else {\n      return null;\n    }\n  }\n}\n\nexport type PossibleNode =\n  | ASTv2.PathExpression\n  | ASTv2.AppendContent\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock;\n\nexport const KEYWORD_NODES = {\n  Call: ['Call'],\n  Block: ['InvokeBlock'],\n  Append: ['AppendContent'],\n  Modifier: ['ElementModifier'],\n} as const;\n\nexport interface KeywordCandidates {\n  Call: ASTv2.ExpressionNode;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordCandidate = KeywordCandidates[keyof KeywordCandidates];\n\nexport interface KeywordMatches {\n  Call: ASTv2.CallExpression;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordMatch = KeywordMatches[keyof KeywordMatches];\n\n/**\n * A \"generic\" keyword is something like `has-block`, which makes sense in the context\n * of sub-expression or append\n */\nexport type GenericKeywordNode = ASTv2.AppendContent | ASTv2.CallExpression;\n\nexport type KeywordNode =\n  | GenericKeywordNode\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock\n  | ASTv2.ElementModifier;\n\nexport function keyword<\n  K extends KeywordType,\n  D extends KeywordDelegate<KeywordMatches[K], unknown, Out>,\n  Out = unknown\n>(keyword: string, type: K, delegate: D): Keyword<K, Out> {\n  return new KeywordImpl(keyword, type, delegate as KeywordDelegate<KeywordMatch, unknown, Out>);\n}\n\nexport type PossibleKeyword = KeywordNode;\ntype OutFor<K extends Keyword | BlockKeyword> = K extends BlockKeyword<infer Out>\n  ? Out\n  : K extends Keyword<KeywordType, infer Out>\n  ? Out\n  : never;\n\nfunction getCalleeExpression(\n  node: KeywordNode | ASTv2.ExpressionNode\n): ASTv2.ExpressionNode | null {\n  switch (node.type) {\n    // This covers the inside of attributes and expressions, as well as the callee\n    // of call nodes\n    case 'Path':\n      return node;\n    case 'AppendContent':\n      return getCalleeExpression(node.value);\n    case 'Call':\n    case 'InvokeBlock':\n    case 'ElementModifier':\n      return node.callee;\n    default:\n      return null;\n  }\n}\n\nexport class Keywords<K extends KeywordType, KeywordList extends Keyword<K> = never>\n  implements Keyword<K, OutFor<KeywordList>> {\n  #keywords: Keyword[] = [];\n  #type: K;\n\n  constructor(type: K) {\n    this.#type = type;\n  }\n\n  kw<S extends string = string, Out = unknown>(\n    name: S,\n    delegate: KeywordDelegate<KeywordMatches[K], unknown, Out>\n  ): Keywords<K, KeywordList | Keyword<K, Out>> {\n    this.#keywords.push(keyword(name, this.#type, delegate));\n\n    return this;\n  }\n\n  translate(\n    node: KeywordCandidates[K],\n    state: NormalizationState\n  ): Result<OutFor<KeywordList>> | null {\n    for (let keyword of this.#keywords) {\n      let result = keyword.translate(node, state) as Result<OutFor<KeywordList>>;\n      if (result !== null) {\n        return result;\n      }\n    }\n\n    let path = getCalleeExpression(node);\n\n    if (path && path.type === 'Path' && path.ref.type === 'Free' && isKeyword(path.ref.name)) {\n      let { name } = path.ref;\n\n      let usedType = this.#type;\n      let validTypes = KEYWORDS_TYPES[name];\n\n      if (validTypes.indexOf(usedType) === -1) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${name}\\` keyword was used incorrectly. It was used as ${\n              typesToReadableName[usedType]\n            }, but its valid usages are:\\n\\n${generateTypesMessage(\n              name,\n              validTypes\n            )}\\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n    }\n\n    return null;\n  }\n}\n\nconst typesToReadableName = {\n  Append: 'an append statement',\n  Block: 'a block statement',\n  Call: 'a call expression',\n  Modifier: 'a modifier',\n};\n\nfunction generateTypesMessage(name: string, types: KeywordType[]): string {\n  return types\n    .map((type) => {\n      switch (type) {\n        case 'Append':\n          return `- As an append statement, as in: {{${name}}}`;\n        case 'Block':\n          return `- As a block statement, as in: {{#${name}}}{{/${name}}}`;\n        case 'Call':\n          return `- As an expression, as in: (${name})`;\n        case 'Modifier':\n          return `- As a modifier, as in: <div {{${name}}}></div>`;\n        default:\n          return exhausted(type);\n      }\n    })\n    .join('\\n\\n');\n}\n\n/**\n * This function builds keyword definitions for a particular type of AST node (`KeywordType`).\n *\n * You can build keyword definitions for:\n *\n * - `Expr`: A `SubExpression` or `PathExpression`\n * - `Block`: A `BlockStatement`\n *   - A `BlockStatement` is a keyword candidate if its head is a\n *     `PathExpression`\n * - `Append`: An `AppendStatement`\n *\n * A node is a keyword candidate if:\n *\n * - A `PathExpression` is a keyword candidate if it has no tail, and its\n *   head expression is a `LocalVarHead` or `FreeVarHead` whose name is\n *   the keyword's name.\n * - A `SubExpression`, `AppendStatement`, or `BlockStatement` is a keyword\n *   candidate if its head is a keyword candidate.\n *\n * The keyword infrastructure guarantees that:\n *\n * - If a node is not a keyword candidate, it is never passed to any keyword's\n *   `assert` method.\n * - If a node is not the `KeywordType` for a particular keyword, it will not\n *   be passed to the keyword's `assert` method.\n *\n * `Expr` keywords are used in expression positions and should return HIR\n * expressions. `Block` and `Append` keywords are used in statement\n * positions and should return HIR statements.\n *\n * A keyword definition has two parts:\n *\n * - `match`, which determines whether an AST node matches the keyword, and can\n *   optionally return some information extracted from the AST node.\n * - `translate`, which takes a matching AST node as well as the extracted\n *   information and returns an appropriate HIR instruction.\n *\n * # Example\n *\n * This keyword:\n *\n * - turns `(hello)` into `\"hello\"`\n *   - as long as `hello` is not in scope\n * - makes it an error to pass any arguments (such as `(hello world)`)\n *\n * ```ts\n * keywords('SubExpr').kw('hello', {\n *   assert(node: ExprKeywordNode): Result<void> | false {\n *     // we don't want to transform `hello` as a `PathExpression`\n *     if (node.type !== 'SubExpression') {\n *       return false;\n *     }\n *\n *     // node.head would be `LocalVarHead` if `hello` was in scope\n *     if (node.head.type !== 'FreeVarHead') {\n *       return false;\n *     }\n *\n *     if (node.params.length || node.hash) {\n *       return Err(generateSyntaxError(`(hello) does not take any arguments`), node.loc);\n *     } else {\n *       return Ok();\n *     }\n *   },\n *\n *   translate(node: ASTv2.SubExpression): hir.Expression {\n *     return ASTv2.builders.literal(\"hello\", node.loc)\n *   }\n * })\n * ```\n *\n * The keyword infrastructure checks to make sure that the node is the right\n * type before calling `assert`, so you only need to consider `SubExpression`\n * and `PathExpression` here. It also checks to make sure that the node passed\n * to `assert` has the keyword name in the right place.\n *\n * Note the important difference between returning `false` from `assert`,\n * which just means that the node didn't match, and returning `Err`, which\n * means that the node matched, but there was a keyword-specific syntax\n * error.\n */\nexport function keywords<K extends KeywordType>(type: K): Keywords<K> {\n  return new Keywords(type);\n}\n"],"sourceRoot":""} |
@@ -1,5 +0,5 @@ | ||
export { precompile, precompileJSON } from './lib/compiler'; | ||
export { defaultId, precompile, precompileJSON } from './lib/compiler'; | ||
export { ProgramSymbols, buildStatement, buildStatements, s, c, unicode, NEWLINE } from './lib/builder/builder'; // exported only for tests | ||
export { default as WireFormatDebugger } from './lib/wire-format-debug'; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3BhY2thZ2VzL0BnbGltbWVyL2NvbXBpbGVyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLFNBQUEsVUFBQSxFQUFBLGNBQUEsUUFBQSxnQkFBQTtBQUNBLFNBQUEsY0FBQSxFQUFBLGNBQUEsRUFBQSxlQUFBLEVBQUEsQ0FBQSxFQUFBLENBQUEsRUFBQSxPQUFBLEVBQUEsT0FBQSxRQUFBLHVCQUFBLEMsQ0FXQTs7QUFDQSxTQUFTLE9BQU8sSUFBaEIsa0JBQUEsUUFBQSx5QkFBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IHByZWNvbXBpbGUsIHByZWNvbXBpbGVKU09OLCBQcmVjb21waWxlT3B0aW9ucyB9IGZyb20gJy4vbGliL2NvbXBpbGVyJztcbmV4cG9ydCB7XG4gIFByb2dyYW1TeW1ib2xzLFxuICBidWlsZFN0YXRlbWVudCxcbiAgYnVpbGRTdGF0ZW1lbnRzLFxuICBzLFxuICBjLFxuICB1bmljb2RlLFxuICBORVdMSU5FLFxufSBmcm9tICcuL2xpYi9idWlsZGVyL2J1aWxkZXInO1xuZXhwb3J0IHsgQnVpbGRlclN0YXRlbWVudCwgQnVpbGRlciB9IGZyb20gJy4vbGliL2J1aWxkZXIvYnVpbGRlci1pbnRlcmZhY2UnO1xuXG4vLyBleHBvcnRlZCBvbmx5IGZvciB0ZXN0c1xuZXhwb3J0IHsgZGVmYXVsdCBhcyBXaXJlRm9ybWF0RGVidWdnZXIgfSBmcm9tICcuL2xpYi93aXJlLWZvcm1hdC1kZWJ1Zyc7XG4iXSwic291cmNlUm9vdCI6IiJ9 | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3BhY2thZ2VzL0BnbGltbWVyL2NvbXBpbGVyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLFNBQUEsU0FBQSxFQUFBLFVBQUEsRUFBQSxjQUFBLFFBQUEsZ0JBQUE7QUFDQSxTQUFBLGNBQUEsRUFBQSxjQUFBLEVBQUEsZUFBQSxFQUFBLENBQUEsRUFBQSxDQUFBLEVBQUEsT0FBQSxFQUFBLE9BQUEsUUFBQSx1QkFBQSxDLENBV0E7O0FBQ0EsU0FBUyxPQUFPLElBQWhCLGtCQUFBLFFBQUEseUJBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgeyBkZWZhdWx0SWQsIHByZWNvbXBpbGUsIHByZWNvbXBpbGVKU09OLCBQcmVjb21waWxlT3B0aW9ucyB9IGZyb20gJy4vbGliL2NvbXBpbGVyJztcbmV4cG9ydCB7XG4gIFByb2dyYW1TeW1ib2xzLFxuICBidWlsZFN0YXRlbWVudCxcbiAgYnVpbGRTdGF0ZW1lbnRzLFxuICBzLFxuICBjLFxuICB1bmljb2RlLFxuICBORVdMSU5FLFxufSBmcm9tICcuL2xpYi9idWlsZGVyL2J1aWxkZXInO1xuZXhwb3J0IHsgQnVpbGRlclN0YXRlbWVudCwgQnVpbGRlciB9IGZyb20gJy4vbGliL2J1aWxkZXIvYnVpbGRlci1pbnRlcmZhY2UnO1xuXG4vLyBleHBvcnRlZCBvbmx5IGZvciB0ZXN0c1xuZXhwb3J0IHsgZGVmYXVsdCBhcyBXaXJlRm9ybWF0RGVidWdnZXIgfSBmcm9tICcuL2xpYi93aXJlLWZvcm1hdC1kZWJ1Zyc7XG4iXSwic291cmNlUm9vdCI6IiJ9 |
@@ -5,2 +5,28 @@ import { normalize, Source } from '@glimmer/syntax'; | ||
import { visit as pass2 } from './passes/2-encoding/index'; | ||
export var defaultId = function () { | ||
var req = typeof module === 'object' && typeof module.require === 'function' ? module.require : require; | ||
if (req) { | ||
try { | ||
var crypto = req('crypto'); | ||
var idFn = function idFn(src) { | ||
var hash = crypto.createHash('sha1'); | ||
hash.update(src, 'utf8'); // trim to 6 bytes of data (2^48 - 1) | ||
return hash.digest('base64').substring(0, 8); | ||
}; | ||
idFn('test'); | ||
return idFn; | ||
} catch (e) {} | ||
} | ||
return function idFn() { | ||
return null; | ||
}; | ||
}(); | ||
var defaultOptions = { | ||
id: defaultId | ||
}; | ||
/* | ||
@@ -23,3 +49,3 @@ * Compile a string into a template javascript string. | ||
if (options === void 0) { | ||
options = {}; | ||
options = defaultOptions; | ||
} | ||
@@ -71,3 +97,3 @@ | ||
if (options === void 0) { | ||
options = {}; | ||
options = defaultOptions; | ||
} | ||
@@ -82,4 +108,6 @@ | ||
var moduleName = (_a = options.meta) === null || _a === void 0 ? void 0 : _a.moduleName; | ||
var idFn = options.id || defaultId; | ||
var blockJSON = JSON.stringify(block); | ||
var templateJSONObject = { | ||
id: idFn(JSON.stringify(options.meta) + blockJSON), | ||
block: blockJSON, | ||
@@ -109,2 +137,2 @@ moduleName: moduleName !== null && moduleName !== void 0 ? moduleName : '(unknown template module)', | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3BhY2thZ2VzL0BnbGltbWVyL2NvbXBpbGVyL2xpYi9jb21waWxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFNQSxTQUFBLFNBQUEsRUFBQSxNQUFBLFFBQUEsaUJBQUE7QUFDQSxTQUFBLFlBQUEsUUFBQSxlQUFBO0FBRUEsT0FBQSxLQUFBLE1BQUEsZ0NBQUE7QUFDQSxTQUFTLEtBQUssSUFBZCxLQUFBLFFBQUEsMkJBQUE7QUFFQTs7Ozs7Ozs7Ozs7Ozs7O0FBY0EsT0FBTSxTQUFBLGNBQUEsQ0FBQSxNQUFBLEVBRUosT0FGSSxFQUUyQjtBQUFBLE1BQS9CLE9BQStCO0FBQS9CLElBQUEsT0FBK0IsR0FGM0IsRUFFMkI7QUFBQTs7OztBQUUvQixNQUFJLE1BQU0sR0FBRyxJQUFBLE1BQUEsQ0FBQSxNQUFBLEVBQWlCLENBQUEsRUFBQSxHQUFFLE9BQU8sQ0FBVCxJQUFBLE1BQUEsSUFBQSxJQUFjLEVBQUEsS0FBQSxLQUFkLENBQUEsR0FBYyxLQUFkLENBQUEsR0FBYyxFQUFBLENBQTVDLFVBQWEsQ0FBYjs7QUFGK0IsbUJBR1gsU0FBUyxDQUFBLE1BQUEsRUFBN0IsT0FBNkIsQ0FIRTtBQUFBLE1BRzNCLEdBSDJCO0FBQUEsTUFHM0IsTUFIMkI7O0FBSS9CLE1BQUksS0FBSyxHQUFHLEtBQUssQ0FBQSxNQUFBLEVBQUEsR0FBQSxFQUFZLENBQUEsRUFBQSxHQUFFLE9BQU8sQ0FBVCxVQUFBLE1BQUEsSUFBQSxJQUFvQixFQUFBLEtBQUEsS0FBcEIsQ0FBQSxHQUFBLEVBQUEsR0FBakIsS0FBSyxDQUFMLENBQUEsS0FBQSxDQUF1RCxVQUFBLE9BQUQsRUFBWTtBQUM1RSxXQUFPLEtBQUssQ0FBWixPQUFZLENBQVo7QUFERixHQUFZLENBQVo7O0FBSUEsTUFBQTtBQUFBO0FBQUEsSUFBc0I7QUFDcEIsTUFBQSxZQUFZLENBQVosR0FBQSxnQkFBQSxLQUFBO0FBQ0Q7O0FBRUQsTUFBSSxLQUFLLENBQVQsSUFBQSxFQUFnQjtBQUNkLFdBQU8sQ0FBQyxLQUFLLENBQU4sS0FBQSxFQUFQLE1BQU8sQ0FBUDtBQURGLEdBQUEsTUFFTztBQUNMLFVBQU0sS0FBSyxDQUFYLE1BQUE7QUFDRDtFQUdIO0FBQ0E7O0FBQ0EsSUFBTSxpQkFBaUIsR0FBdkIsc0NBQUE7QUFFQTs7Ozs7Ozs7Ozs7Ozs7O0FBY0EsT0FBTSxTQUFBLFVBQUEsQ0FBQSxNQUFBLEVBQXFDLE9BQXJDLEVBQW9FO0FBQUEsTUFBL0IsT0FBK0I7QUFBL0IsSUFBQSxPQUErQixHQUFwRSxFQUFvRTtBQUFBOzs7O0FBQUEsd0JBQzlDLGNBQWMsQ0FBQSxNQUFBLEVBQXhDLE9BQXdDLENBRGdDO0FBQUEsTUFDcEUsS0FEb0U7QUFBQSxNQUNwRSxVQURvRTs7QUFHeEUsTUFBSSxVQUFVLEdBQUEsQ0FBQSxFQUFBLEdBQUcsT0FBTyxDQUFWLElBQUEsTUFBQSxJQUFBLElBQWUsRUFBQSxLQUFBLEtBQWYsQ0FBQSxHQUFlLEtBQWYsQ0FBQSxHQUFlLEVBQUEsQ0FBN0IsVUFBQTtBQUNBLE1BQUksU0FBUyxHQUFHLElBQUksQ0FBSixTQUFBLENBQWhCLEtBQWdCLENBQWhCO0FBQ0EsTUFBSSxrQkFBa0IsR0FBb0M7QUFDeEQsSUFBQSxLQUFLLEVBRG1ELFNBQUE7QUFFeEQsSUFBQSxVQUFVLEVBQUUsVUFBVSxLQUFWLElBQUEsSUFBQSxVQUFVLEtBQUEsS0FBVixDQUFBLEdBQUEsVUFBQSxHQUY0QywyQkFBQTtBQUd4RDtBQUNBO0FBQ0EsSUFBQSxLQUFLLEVBTG1ELGlCQUFBO0FBTXhELElBQUEsWUFBWSxFQUFBLENBQUEsRUFBQSxHQUFFLE9BQU8sQ0FBVCxVQUFBLE1BQUEsSUFBQSxJQUFvQixFQUFBLEtBQUEsS0FBcEIsQ0FBQSxHQUFBLEVBQUEsR0FBd0I7QUFOb0IsR0FBMUQ7O0FBU0EsTUFBSSxDQUFDLE9BQU8sQ0FBWixVQUFBLEVBQXlCO0FBQ3ZCLFdBQU8sa0JBQWtCLENBQXpCLEtBQUE7QUFmc0UsR0FBQSxDQWtCeEU7OztBQUNBLE1BQUksV0FBVyxHQUFHLElBQUksQ0FBSixTQUFBLENBQWxCLGtCQUFrQixDQUFsQjs7QUFFQSxNQUFJLE9BQU8sQ0FBUCxVQUFBLElBQXNCLFVBQVUsQ0FBVixNQUFBLEdBQTFCLENBQUEsRUFBaUQ7QUFDL0MsUUFBSSxPQUFPLGFBQVcsVUFBVSxDQUFWLElBQUEsQ0FBdEIsR0FBc0IsQ0FBWCxNQUFYO0FBRUEsSUFBQSxXQUFXLEdBQUcsV0FBVyxDQUFYLE9BQUEsUUFBQSxpQkFBQSxTQUFkLE9BQWMsQ0FBZDtBQUhGLEdBQUEsTUFJTztBQUNMLElBQUEsV0FBVyxHQUFHLFdBQVcsQ0FBWCxPQUFBLFFBQUEsaUJBQUEsU0FBZCxNQUFjLENBQWQ7QUFDRDs7QUFFRCxTQUFBLFdBQUE7QUFDRCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIFNlcmlhbGl6ZWRUZW1wbGF0ZUJsb2NrLFxuICBTZXJpYWxpemVkVGVtcGxhdGVXaXRoTGF6eUJsb2NrLFxuICBUZW1wbGF0ZUphdmFzY3JpcHQsXG59IGZyb20gJ0BnbGltbWVyL2ludGVyZmFjZXMnO1xuaW1wb3J0IHsgTE9DQUxfU0hPVUxEX0xPRyB9IGZyb20gJ0BnbGltbWVyL2xvY2FsLWRlYnVnLWZsYWdzJztcbmltcG9ydCB7IG5vcm1hbGl6ZSwgUHJlY29tcGlsZU9wdGlvbnMsIFNvdXJjZSB9IGZyb20gJ0BnbGltbWVyL3N5bnRheCc7XG5pbXBvcnQgeyBMT0NBTF9MT0dHRVIgfSBmcm9tICdAZ2xpbW1lci91dGlsJztcblxuaW1wb3J0IHBhc3MwIGZyb20gJy4vcGFzc2VzLzEtbm9ybWFsaXphdGlvbi9pbmRleCc7XG5pbXBvcnQgeyB2aXNpdCBhcyBwYXNzMiB9IGZyb20gJy4vcGFzc2VzLzItZW5jb2RpbmcvaW5kZXgnO1xuXG4vKlxuICogQ29tcGlsZSBhIHN0cmluZyBpbnRvIGEgdGVtcGxhdGUgamF2YXNjcmlwdCBzdHJpbmcuXG4gKlxuICogRXhhbXBsZSB1c2FnZTpcbiAqICAgICBpbXBvcnQgeyBwcmVjb21waWxlIH0gZnJvbSAnQGdsaW1tZXIvY29tcGlsZXInO1xuICogICAgIGltcG9ydCB7IHRlbXBsYXRlRmFjdG9yeSB9IGZyb20gJ2dsaW1tZXItcnVudGltZSc7XG4gKiAgICAgbGV0IHRlbXBsYXRlSnMgPSBwcmVjb21waWxlKFwiSG93ZHkge3tuYW1lfX1cIik7XG4gKiAgICAgbGV0IGZhY3RvcnkgPSB0ZW1wbGF0ZUZhY3RvcnkobmV3IEZ1bmN0aW9uKFwicmV0dXJuIFwiICsgdGVtcGxhdGVKcykoKSk7XG4gKiAgICAgbGV0IHRlbXBsYXRlID0gZmFjdG9yeS5jcmVhdGUoZW52KTtcbiAqXG4gKiBAbWV0aG9kIHByZWNvbXBpbGVcbiAqIEBwYXJhbSB7c3RyaW5nfSBzdHJpbmcgYSBHbGltbWVyIHRlbXBsYXRlIHN0cmluZ1xuICogQHJldHVybiB7c3RyaW5nfSBhIHRlbXBsYXRlIGphdmFzY3JpcHQgc3RyaW5nXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBwcmVjb21waWxlSlNPTihcbiAgc3RyaW5nOiBzdHJpbmcsXG4gIG9wdGlvbnM6IFByZWNvbXBpbGVPcHRpb25zID0ge31cbik6IFtibG9jazogU2VyaWFsaXplZFRlbXBsYXRlQmxvY2ssIHVzZWRMb2NhbHM6IHN0cmluZ1tdXSB7XG4gIGxldCBzb3VyY2UgPSBuZXcgU291cmNlKHN0cmluZywgb3B0aW9ucy5tZXRhPy5tb2R1bGVOYW1lKTtcbiAgbGV0IFthc3QsIGxvY2Fsc10gPSBub3JtYWxpemUoc291cmNlLCBvcHRpb25zKTtcbiAgbGV0IGJsb2NrID0gcGFzczAoc291cmNlLCBhc3QsIG9wdGlvbnMuc3RyaWN0TW9kZSA/PyBmYWxzZSkubWFwT2soKHBhc3MySW4pID0+IHtcbiAgICByZXR1cm4gcGFzczIocGFzczJJbik7XG4gIH0pO1xuXG4gIGlmIChMT0NBTF9TSE9VTERfTE9HKSB7XG4gICAgTE9DQUxfTE9HR0VSLmxvZyhgVGVtcGxhdGUgLT5gLCBibG9jayk7XG4gIH1cblxuICBpZiAoYmxvY2suaXNPaykge1xuICAgIHJldHVybiBbYmxvY2sudmFsdWUsIGxvY2Fsc107XG4gIH0gZWxzZSB7XG4gICAgdGhyb3cgYmxvY2sucmVhc29uO1xuICB9XG59XG5cbi8vIFVVSUQgdXNlZCBhcyBhIHVuaXF1ZSBwbGFjZWhvbGRlciBmb3IgcGxhY2luZyBhIHNuaXBwZXQgb2YgSlMgY29kZSBpbnRvXG4vLyB0aGUgb3RoZXJ3aXNlIEpTT04gc3RyaW5naWZpZWQgdmFsdWUgYmVsb3cuXG5jb25zdCBTQ09QRV9QTEFDRUhPTERFUiA9ICc3OTZkMjRlNi0yNDUwLTRmYjAtOGNkZi1iNjU2MzhiNWVmNzAnO1xuXG4vKlxuICogQ29tcGlsZSBhIHN0cmluZyBpbnRvIGEgdGVtcGxhdGUgamF2YXNjcmlwdCBzdHJpbmcuXG4gKlxuICogRXhhbXBsZSB1c2FnZTpcbiAqICAgICBpbXBvcnQgeyBwcmVjb21waWxlIH0gZnJvbSAnQGdsaW1tZXIvY29tcGlsZXInO1xuICogICAgIGltcG9ydCB7IHRlbXBsYXRlRmFjdG9yeSB9IGZyb20gJ2dsaW1tZXItcnVudGltZSc7XG4gKiAgICAgbGV0IHRlbXBsYXRlSnMgPSBwcmVjb21waWxlKFwiSG93ZHkge3tuYW1lfX1cIik7XG4gKiAgICAgbGV0IGZhY3RvcnkgPSB0ZW1wbGF0ZUZhY3RvcnkobmV3IEZ1bmN0aW9uKFwicmV0dXJuIFwiICsgdGVtcGxhdGVKcykoKSk7XG4gKiAgICAgbGV0IHRlbXBsYXRlID0gZmFjdG9yeS5jcmVhdGUoZW52KTtcbiAqXG4gKiBAbWV0aG9kIHByZWNvbXBpbGVcbiAqIEBwYXJhbSB7c3RyaW5nfSBzdHJpbmcgYSBHbGltbWVyIHRlbXBsYXRlIHN0cmluZ1xuICogQHJldHVybiB7c3RyaW5nfSBhIHRlbXBsYXRlIGphdmFzY3JpcHQgc3RyaW5nXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBwcmVjb21waWxlKHNvdXJjZTogc3RyaW5nLCBvcHRpb25zOiBQcmVjb21waWxlT3B0aW9ucyA9IHt9KTogVGVtcGxhdGVKYXZhc2NyaXB0IHtcbiAgbGV0IFtibG9jaywgdXNlZExvY2Fsc10gPSBwcmVjb21waWxlSlNPTihzb3VyY2UsIG9wdGlvbnMpO1xuXG4gIGxldCBtb2R1bGVOYW1lID0gb3B0aW9ucy5tZXRhPy5tb2R1bGVOYW1lO1xuICBsZXQgYmxvY2tKU09OID0gSlNPTi5zdHJpbmdpZnkoYmxvY2spO1xuICBsZXQgdGVtcGxhdGVKU09OT2JqZWN0OiBTZXJpYWxpemVkVGVtcGxhdGVXaXRoTGF6eUJsb2NrID0ge1xuICAgIGJsb2NrOiBibG9ja0pTT04sXG4gICAgbW9kdWxlTmFtZTogbW9kdWxlTmFtZSA/PyAnKHVua25vd24gdGVtcGxhdGUgbW9kdWxlKScsXG4gICAgLy8gbHlpbmcgdG8gdGhlIHR5cGUgY2hlY2tlciBoZXJlIGJlY2F1c2Ugd2UncmUgZ29pbmcgdG9cbiAgICAvLyByZXBsYWNlIGl0IGp1c3QgYmVsb3csIGFmdGVyIHN0cmluZ2lmaWNhdGlvblxuICAgIHNjb3BlOiAoU0NPUEVfUExBQ0VIT0xERVIgYXMgdW5rbm93bikgYXMgbnVsbCxcbiAgICBpc1N0cmljdE1vZGU6IG9wdGlvbnMuc3RyaWN0TW9kZSA/PyBmYWxzZSxcbiAgfTtcblxuICBpZiAoIW9wdGlvbnMuc3RyaWN0TW9kZSkge1xuICAgIGRlbGV0ZSB0ZW1wbGF0ZUpTT05PYmplY3Quc2NvcGU7XG4gIH1cblxuICAvLyBKU09OIGlzIGphdmFzY3JpcHRcbiAgbGV0IHN0cmluZ2lmaWVkID0gSlNPTi5zdHJpbmdpZnkodGVtcGxhdGVKU09OT2JqZWN0KTtcblxuICBpZiAob3B0aW9ucy5zdHJpY3RNb2RlICYmIHVzZWRMb2NhbHMubGVuZ3RoID4gMCkge1xuICAgIGxldCBzY29wZUZuID0gYCgpPT5bJHt1c2VkTG9jYWxzLmpvaW4oJywnKX1dYDtcblxuICAgIHN0cmluZ2lmaWVkID0gc3RyaW5naWZpZWQucmVwbGFjZShgXCIke1NDT1BFX1BMQUNFSE9MREVSfVwiYCwgc2NvcGVGbik7XG4gIH0gZWxzZSB7XG4gICAgc3RyaW5naWZpZWQgPSBzdHJpbmdpZmllZC5yZXBsYWNlKGBcIiR7U0NPUEVfUExBQ0VIT0xERVJ9XCJgLCAnbnVsbCcpO1xuICB9XG5cbiAgcmV0dXJuIHN0cmluZ2lmaWVkO1xufVxuXG5leHBvcnQgeyBQcmVjb21waWxlT3B0aW9ucyB9O1xuIl0sInNvdXJjZVJvb3QiOiIifQ== | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../packages/@glimmer/compiler/lib/compiler.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  SerializedTemplateBlock,\n  SerializedTemplateWithLazyBlock,\n  TemplateJavascript,\n} from '@glimmer/interfaces';\nimport { LOCAL_SHOULD_LOG } from '@glimmer/local-debug-flags';\nimport { normalize, PrecompileOptions, Source, TemplateIdFn } from '@glimmer/syntax';\nimport { LOCAL_LOGGER } from '@glimmer/util';\n\nimport pass0 from './passes/1-normalization/index';\nimport { visit as pass2 } from './passes/2-encoding/index';\n\ndeclare function require(id: 'crypto'): Crypto;\ndeclare function require(id: string): unknown;\n\ninterface Crypto {\n  createHash(\n    alg: 'sha1'\n  ): {\n    update(src: string, encoding: 'utf8'): void;\n    digest(encoding: 'base64'): string;\n  };\n}\n\nexport const defaultId: TemplateIdFn = (() => {\n  let req: typeof require | undefined =\n    typeof module === 'object' && typeof module.require === 'function' ? module.require : require;\n\n  if (req) {\n    try {\n      const crypto = req('crypto');\n\n      let idFn: TemplateIdFn = (src) => {\n        let hash = crypto.createHash('sha1');\n        hash.update(src, 'utf8');\n        // trim to 6 bytes of data (2^48 - 1)\n        return hash.digest('base64').substring(0, 8);\n      };\n\n      idFn('test');\n\n      return idFn;\n    } catch (e) {}\n  }\n\n  return function idFn() {\n    return null;\n  };\n})();\n\nconst defaultOptions: PrecompileOptions = {\n  id: defaultId,\n};\n\n/*\n * Compile a string into a template javascript string.\n *\n * Example usage:\n *     import { precompile } from '@glimmer/compiler';\n *     import { templateFactory } from 'glimmer-runtime';\n *     let templateJs = precompile(\"Howdy {{name}}\");\n *     let factory = templateFactory(new Function(\"return \" + templateJs)());\n *     let template = factory.create(env);\n *\n * @method precompile\n * @param {string} string a Glimmer template string\n * @return {string} a template javascript string\n */\nexport function precompileJSON(\n  string: string,\n  options: PrecompileOptions = defaultOptions\n): [block: SerializedTemplateBlock, usedLocals: string[]] {\n  let source = new Source(string, options.meta?.moduleName);\n  let [ast, locals] = normalize(source, options);\n  let block = pass0(source, ast, options.strictMode ?? false).mapOk((pass2In) => {\n    return pass2(pass2In);\n  });\n\n  if (LOCAL_SHOULD_LOG) {\n    LOCAL_LOGGER.log(`Template ->`, block);\n  }\n\n  if (block.isOk) {\n    return [block.value, locals];\n  } else {\n    throw block.reason;\n  }\n}\n\n// UUID used as a unique placeholder for placing a snippet of JS code into\n// the otherwise JSON stringified value below.\nconst SCOPE_PLACEHOLDER = '796d24e6-2450-4fb0-8cdf-b65638b5ef70';\n\n/*\n * Compile a string into a template javascript string.\n *\n * Example usage:\n *     import { precompile } from '@glimmer/compiler';\n *     import { templateFactory } from 'glimmer-runtime';\n *     let templateJs = precompile(\"Howdy {{name}}\");\n *     let factory = templateFactory(new Function(\"return \" + templateJs)());\n *     let template = factory.create(env);\n *\n * @method precompile\n * @param {string} string a Glimmer template string\n * @return {string} a template javascript string\n */\nexport function precompile(\n  source: string,\n  options: PrecompileOptions = defaultOptions\n): TemplateJavascript {\n  let [block, usedLocals] = precompileJSON(source, options);\n\n  let moduleName = options.meta?.moduleName;\n  let idFn = options.id || defaultId;\n  let blockJSON = JSON.stringify(block);\n  let templateJSONObject: SerializedTemplateWithLazyBlock = {\n    id: idFn(JSON.stringify(options.meta) + blockJSON),\n    block: blockJSON,\n    moduleName: moduleName ?? '(unknown template module)',\n    // lying to the type checker here because we're going to\n    // replace it just below, after stringification\n    scope: (SCOPE_PLACEHOLDER as unknown) as null,\n    isStrictMode: options.strictMode ?? false,\n  };\n\n  if (!options.strictMode) {\n    delete templateJSONObject.scope;\n  }\n\n  // JSON is javascript\n  let stringified = JSON.stringify(templateJSONObject);\n\n  if (options.strictMode && usedLocals.length > 0) {\n    let scopeFn = `()=>[${usedLocals.join(',')}]`;\n\n    stringified = stringified.replace(`\"${SCOPE_PLACEHOLDER}\"`, scopeFn);\n  } else {\n    stringified = stringified.replace(`\"${SCOPE_PLACEHOLDER}\"`, 'null');\n  }\n\n  return stringified;\n}\n\nexport { PrecompileOptions };\n"],"sourceRoot":""} |
@@ -51,5 +51,5 @@ function _createForOfIteratorHelperLoose(o, allowArrayLike) { var it; if (typeof Symbol === "undefined" || o[Symbol.iterator] == null) { if (Array.isArray(o) || (it = _unsupportedIterableToArray(o)) || allowArrayLike && o && typeof o.length === "number") { if (it) o = it; var i = 0; return function () { if (i >= o.length) return { done: true }; return { done: false, value: o[i++] }; }; } throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } it = o[Symbol.iterator](); return it.next.bind(it); } | ||
var path = getPathExpression(node); | ||
var path = getCalleeExpression(node); | ||
if (path !== null && path.ref.type === 'Free') { | ||
if (path !== null && path.type === 'Path' && path.ref.type === 'Free') { | ||
if (path.tail.length > 0) { | ||
@@ -73,5 +73,5 @@ if (path.ref.resolution.serialize() === 'Loose') { | ||
if (this.match(node)) { | ||
var path = getPathExpression(node); | ||
var path = getCalleeExpression(node); | ||
if (path !== null && path.tail.length > 0) { | ||
if (path !== null && path.type === 'Path' && path.tail.length > 0) { | ||
return Err(generateSyntaxError("The `" + this.keyword + "` keyword was used incorrectly. It was used as `" + path.loc.asString() + "`, but it cannot be used with additional path segments. \n\nError caused by", node.loc)); | ||
@@ -105,3 +105,3 @@ } | ||
function getPathExpression(node) { | ||
function getCalleeExpression(node) { | ||
switch (node.type) { | ||
@@ -114,3 +114,3 @@ // This covers the inside of attributes and expressions, as well as the callee | ||
case 'AppendContent': | ||
return getPathExpression(node.value); | ||
return getCalleeExpression(node.value); | ||
@@ -120,3 +120,3 @@ case 'Call': | ||
case 'ElementModifier': | ||
return getPathExpression(node.callee); | ||
return node.callee; | ||
@@ -156,5 +156,5 @@ default: | ||
var path = getPathExpression(node); | ||
var path = getCalleeExpression(node); | ||
if (path && path.ref.type === 'Free' && isKeyword(path.ref.name)) { | ||
if (path && path.type === 'Path' && path.ref.type === 'Free' && isKeyword(path.ref.name)) { | ||
var name = path.ref.name; | ||
@@ -290,2 +290,2 @@ | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../../../packages/@glimmer/compiler/lib/passes/1-normalization/keywords/impl.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ASTv2,\n  generateSyntaxError,\n  isKeyword,\n  KEYWORDS_TYPES,\n  KeywordType,\n} from '@glimmer/syntax';\nimport { exhausted } from '@glimmer/util';\n\nimport { Err, Result } from '../../../shared/result';\nimport { NormalizationState } from '../context';\n\nexport interface KeywordDelegate<Match extends KeywordMatch, V, Out> {\n  assert(options: Match, state: NormalizationState): Result<V>;\n  translate(options: { node: Match; state: NormalizationState }, param: V): Result<Out>;\n}\n\nexport interface Keyword<K extends KeywordType = KeywordType, Out = unknown> {\n  translate(node: KeywordCandidates[K], state: NormalizationState): Result<Out> | null;\n}\n\nexport interface BlockKeyword<Out = unknown> {\n  translate(node: ASTv2.InvokeBlock, state: NormalizationState): Result<Out> | null;\n}\n\nclass KeywordImpl<\n  K extends KeywordType,\n  S extends string = string,\n  Param = unknown,\n  Out = unknown\n> {\n  protected types: Set<KeywordCandidates[K]['type']>;\n\n  constructor(\n    protected keyword: S,\n    type: KeywordType,\n    private delegate: KeywordDelegate<KeywordMatches[K], Param, Out>\n  ) {\n    let nodes = new Set<KeywordNode['type']>();\n    for (let nodeType of KEYWORD_NODES[type]) {\n      nodes.add(nodeType);\n    }\n\n    this.types = nodes;\n  }\n\n  protected match(node: KeywordCandidates[K]): node is KeywordMatches[K] {\n    if (!this.types.has(node.type)) {\n      return false;\n    }\n\n    let path = getPathExpression(node);\n\n    if (path !== null && path.ref.type === 'Free') {\n      if (path.tail.length > 0) {\n        if (path.ref.resolution.serialize() === 'Loose') {\n          // cannot be a keyword reference, keywords do not allow paths (must be\n          // relying on implicit this fallback)\n          return false;\n        }\n      }\n\n      return path.ref.name === this.keyword;\n    } else {\n      return false;\n    }\n  }\n\n  translate(node: KeywordMatches[K], state: NormalizationState): Result<Out> | null {\n    if (this.match(node)) {\n      let path = getPathExpression(node);\n\n      if (path !== null && path.tail.length > 0) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${\n              this.keyword\n            }\\` keyword was used incorrectly. It was used as \\`${path.loc.asString()}\\`, but it cannot be used with additional path segments. \\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n\n      let param = this.delegate.assert(node, state);\n      return param.andThen((param) => this.delegate.translate({ node, state }, param));\n    } else {\n      return null;\n    }\n  }\n}\n\nexport type PossibleNode =\n  | ASTv2.PathExpression\n  | ASTv2.AppendContent\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock;\n\nexport const KEYWORD_NODES = {\n  Call: ['Call'],\n  Block: ['InvokeBlock'],\n  Append: ['AppendContent'],\n  Modifier: ['ElementModifier'],\n} as const;\n\nexport interface KeywordCandidates {\n  Call: ASTv2.ExpressionNode;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordCandidate = KeywordCandidates[keyof KeywordCandidates];\n\nexport interface KeywordMatches {\n  Call: ASTv2.CallExpression;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordMatch = KeywordMatches[keyof KeywordMatches];\n\n/**\n * A \"generic\" keyword is something like `has-block`, which makes sense in the context\n * of sub-expression or append\n */\nexport type GenericKeywordNode = ASTv2.AppendContent | ASTv2.CallExpression;\n\nexport type KeywordNode =\n  | GenericKeywordNode\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock\n  | ASTv2.ElementModifier;\n\nexport function keyword<\n  K extends KeywordType,\n  D extends KeywordDelegate<KeywordMatches[K], unknown, Out>,\n  Out = unknown\n>(keyword: string, type: K, delegate: D): Keyword<K, Out> {\n  return new KeywordImpl(keyword, type, delegate as KeywordDelegate<KeywordMatch, unknown, Out>);\n}\n\nexport type PossibleKeyword = KeywordNode;\ntype OutFor<K extends Keyword | BlockKeyword> = K extends BlockKeyword<infer Out>\n  ? Out\n  : K extends Keyword<KeywordType, infer Out>\n  ? Out\n  : never;\n\nfunction getPathExpression(node: KeywordNode | ASTv2.ExpressionNode): ASTv2.PathExpression | null {\n  switch (node.type) {\n    // This covers the inside of attributes and expressions, as well as the callee\n    // of call nodes\n    case 'Path':\n      return node;\n    case 'AppendContent':\n      return getPathExpression(node.value);\n    case 'Call':\n    case 'InvokeBlock':\n    case 'ElementModifier':\n      return getPathExpression(node.callee);\n    default:\n      return null;\n  }\n}\n\nexport class Keywords<K extends KeywordType, KeywordList extends Keyword<K> = never>\n  implements Keyword<K, OutFor<KeywordList>> {\n  #keywords: Keyword[] = [];\n  #type: K;\n\n  constructor(type: K) {\n    this.#type = type;\n  }\n\n  kw<S extends string = string, Out = unknown>(\n    name: S,\n    delegate: KeywordDelegate<KeywordMatches[K], unknown, Out>\n  ): Keywords<K, KeywordList | Keyword<K, Out>> {\n    this.#keywords.push(keyword(name, this.#type, delegate));\n\n    return this;\n  }\n\n  translate(\n    node: KeywordCandidates[K],\n    state: NormalizationState\n  ): Result<OutFor<KeywordList>> | null {\n    for (let keyword of this.#keywords) {\n      let result = keyword.translate(node, state) as Result<OutFor<KeywordList>>;\n      if (result !== null) {\n        return result;\n      }\n    }\n\n    let path = getPathExpression(node);\n\n    if (path && path.ref.type === 'Free' && isKeyword(path.ref.name)) {\n      let { name } = path.ref;\n\n      let usedType = this.#type;\n      let validTypes = KEYWORDS_TYPES[name];\n\n      if (validTypes.indexOf(usedType) === -1) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${name}\\` keyword was used incorrectly. It was used as ${\n              typesToReadableName[usedType]\n            }, but its valid usages are:\\n\\n${generateTypesMessage(\n              name,\n              validTypes\n            )}\\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n    }\n\n    return null;\n  }\n}\n\nconst typesToReadableName = {\n  Append: 'an append statement',\n  Block: 'a block statement',\n  Call: 'a call expression',\n  Modifier: 'a modifier',\n};\n\nfunction generateTypesMessage(name: string, types: KeywordType[]): string {\n  return types\n    .map((type) => {\n      switch (type) {\n        case 'Append':\n          return `- As an append statement, as in: {{${name}}}`;\n        case 'Block':\n          return `- As a block statement, as in: {{#${name}}}{{/${name}}}`;\n        case 'Call':\n          return `- As an expression, as in: (${name})`;\n        case 'Modifier':\n          return `- As a modifier, as in: <div {{${name}}}></div>`;\n        default:\n          return exhausted(type);\n      }\n    })\n    .join('\\n\\n');\n}\n\n/**\n * This function builds keyword definitions for a particular type of AST node (`KeywordType`).\n *\n * You can build keyword definitions for:\n *\n * - `Expr`: A `SubExpression` or `PathExpression`\n * - `Block`: A `BlockStatement`\n *   - A `BlockStatement` is a keyword candidate if its head is a\n *     `PathExpression`\n * - `Append`: An `AppendStatement`\n *\n * A node is a keyword candidate if:\n *\n * - A `PathExpression` is a keyword candidate if it has no tail, and its\n *   head expression is a `LocalVarHead` or `FreeVarHead` whose name is\n *   the keyword's name.\n * - A `SubExpression`, `AppendStatement`, or `BlockStatement` is a keyword\n *   candidate if its head is a keyword candidate.\n *\n * The keyword infrastructure guarantees that:\n *\n * - If a node is not a keyword candidate, it is never passed to any keyword's\n *   `assert` method.\n * - If a node is not the `KeywordType` for a particular keyword, it will not\n *   be passed to the keyword's `assert` method.\n *\n * `Expr` keywords are used in expression positions and should return HIR\n * expressions. `Block` and `Append` keywords are used in statement\n * positions and should return HIR statements.\n *\n * A keyword definition has two parts:\n *\n * - `match`, which determines whether an AST node matches the keyword, and can\n *   optionally return some information extracted from the AST node.\n * - `translate`, which takes a matching AST node as well as the extracted\n *   information and returns an appropriate HIR instruction.\n *\n * # Example\n *\n * This keyword:\n *\n * - turns `(hello)` into `\"hello\"`\n *   - as long as `hello` is not in scope\n * - makes it an error to pass any arguments (such as `(hello world)`)\n *\n * ```ts\n * keywords('SubExpr').kw('hello', {\n *   assert(node: ExprKeywordNode): Result<void> | false {\n *     // we don't want to transform `hello` as a `PathExpression`\n *     if (node.type !== 'SubExpression') {\n *       return false;\n *     }\n *\n *     // node.head would be `LocalVarHead` if `hello` was in scope\n *     if (node.head.type !== 'FreeVarHead') {\n *       return false;\n *     }\n *\n *     if (node.params.length || node.hash) {\n *       return Err(generateSyntaxError(`(hello) does not take any arguments`), node.loc);\n *     } else {\n *       return Ok();\n *     }\n *   },\n *\n *   translate(node: ASTv2.SubExpression): hir.Expression {\n *     return ASTv2.builders.literal(\"hello\", node.loc)\n *   }\n * })\n * ```\n *\n * The keyword infrastructure checks to make sure that the node is the right\n * type before calling `assert`, so you only need to consider `SubExpression`\n * and `PathExpression` here. It also checks to make sure that the node passed\n * to `assert` has the keyword name in the right place.\n *\n * Note the important difference between returning `false` from `assert`,\n * which just means that the node didn't match, and returning `Err`, which\n * means that the node matched, but there was a keyword-specific syntax\n * error.\n */\nexport function keywords<K extends KeywordType>(type: K): Keywords<K> {\n  return new Keywords(type);\n}\n"],"sourceRoot":""} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../../../packages/@glimmer/compiler/lib/passes/1-normalization/keywords/impl.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ASTv2,\n  generateSyntaxError,\n  isKeyword,\n  KEYWORDS_TYPES,\n  KeywordType,\n} from '@glimmer/syntax';\nimport { exhausted } from '@glimmer/util';\n\nimport { Err, Result } from '../../../shared/result';\nimport { NormalizationState } from '../context';\n\nexport interface KeywordDelegate<Match extends KeywordMatch, V, Out> {\n  assert(options: Match, state: NormalizationState): Result<V>;\n  translate(options: { node: Match; state: NormalizationState }, param: V): Result<Out>;\n}\n\nexport interface Keyword<K extends KeywordType = KeywordType, Out = unknown> {\n  translate(node: KeywordCandidates[K], state: NormalizationState): Result<Out> | null;\n}\n\nexport interface BlockKeyword<Out = unknown> {\n  translate(node: ASTv2.InvokeBlock, state: NormalizationState): Result<Out> | null;\n}\n\nclass KeywordImpl<\n  K extends KeywordType,\n  S extends string = string,\n  Param = unknown,\n  Out = unknown\n> {\n  protected types: Set<KeywordCandidates[K]['type']>;\n\n  constructor(\n    protected keyword: S,\n    type: KeywordType,\n    private delegate: KeywordDelegate<KeywordMatches[K], Param, Out>\n  ) {\n    let nodes = new Set<KeywordNode['type']>();\n    for (let nodeType of KEYWORD_NODES[type]) {\n      nodes.add(nodeType);\n    }\n\n    this.types = nodes;\n  }\n\n  protected match(node: KeywordCandidates[K]): node is KeywordMatches[K] {\n    if (!this.types.has(node.type)) {\n      return false;\n    }\n\n    let path = getCalleeExpression(node);\n\n    if (path !== null && path.type === 'Path' && path.ref.type === 'Free') {\n      if (path.tail.length > 0) {\n        if (path.ref.resolution.serialize() === 'Loose') {\n          // cannot be a keyword reference, keywords do not allow paths (must be\n          // relying on implicit this fallback)\n          return false;\n        }\n      }\n\n      return path.ref.name === this.keyword;\n    } else {\n      return false;\n    }\n  }\n\n  translate(node: KeywordMatches[K], state: NormalizationState): Result<Out> | null {\n    if (this.match(node)) {\n      let path = getCalleeExpression(node);\n\n      if (path !== null && path.type === 'Path' && path.tail.length > 0) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${\n              this.keyword\n            }\\` keyword was used incorrectly. It was used as \\`${path.loc.asString()}\\`, but it cannot be used with additional path segments. \\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n\n      let param = this.delegate.assert(node, state);\n      return param.andThen((param) => this.delegate.translate({ node, state }, param));\n    } else {\n      return null;\n    }\n  }\n}\n\nexport type PossibleNode =\n  | ASTv2.PathExpression\n  | ASTv2.AppendContent\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock;\n\nexport const KEYWORD_NODES = {\n  Call: ['Call'],\n  Block: ['InvokeBlock'],\n  Append: ['AppendContent'],\n  Modifier: ['ElementModifier'],\n} as const;\n\nexport interface KeywordCandidates {\n  Call: ASTv2.ExpressionNode;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordCandidate = KeywordCandidates[keyof KeywordCandidates];\n\nexport interface KeywordMatches {\n  Call: ASTv2.CallExpression;\n  Block: ASTv2.InvokeBlock;\n  Append: ASTv2.AppendContent;\n  Modifier: ASTv2.ElementModifier;\n}\n\nexport type KeywordMatch = KeywordMatches[keyof KeywordMatches];\n\n/**\n * A \"generic\" keyword is something like `has-block`, which makes sense in the context\n * of sub-expression or append\n */\nexport type GenericKeywordNode = ASTv2.AppendContent | ASTv2.CallExpression;\n\nexport type KeywordNode =\n  | GenericKeywordNode\n  | ASTv2.CallExpression\n  | ASTv2.InvokeBlock\n  | ASTv2.ElementModifier;\n\nexport function keyword<\n  K extends KeywordType,\n  D extends KeywordDelegate<KeywordMatches[K], unknown, Out>,\n  Out = unknown\n>(keyword: string, type: K, delegate: D): Keyword<K, Out> {\n  return new KeywordImpl(keyword, type, delegate as KeywordDelegate<KeywordMatch, unknown, Out>);\n}\n\nexport type PossibleKeyword = KeywordNode;\ntype OutFor<K extends Keyword | BlockKeyword> = K extends BlockKeyword<infer Out>\n  ? Out\n  : K extends Keyword<KeywordType, infer Out>\n  ? Out\n  : never;\n\nfunction getCalleeExpression(\n  node: KeywordNode | ASTv2.ExpressionNode\n): ASTv2.ExpressionNode | null {\n  switch (node.type) {\n    // This covers the inside of attributes and expressions, as well as the callee\n    // of call nodes\n    case 'Path':\n      return node;\n    case 'AppendContent':\n      return getCalleeExpression(node.value);\n    case 'Call':\n    case 'InvokeBlock':\n    case 'ElementModifier':\n      return node.callee;\n    default:\n      return null;\n  }\n}\n\nexport class Keywords<K extends KeywordType, KeywordList extends Keyword<K> = never>\n  implements Keyword<K, OutFor<KeywordList>> {\n  #keywords: Keyword[] = [];\n  #type: K;\n\n  constructor(type: K) {\n    this.#type = type;\n  }\n\n  kw<S extends string = string, Out = unknown>(\n    name: S,\n    delegate: KeywordDelegate<KeywordMatches[K], unknown, Out>\n  ): Keywords<K, KeywordList | Keyword<K, Out>> {\n    this.#keywords.push(keyword(name, this.#type, delegate));\n\n    return this;\n  }\n\n  translate(\n    node: KeywordCandidates[K],\n    state: NormalizationState\n  ): Result<OutFor<KeywordList>> | null {\n    for (let keyword of this.#keywords) {\n      let result = keyword.translate(node, state) as Result<OutFor<KeywordList>>;\n      if (result !== null) {\n        return result;\n      }\n    }\n\n    let path = getCalleeExpression(node);\n\n    if (path && path.type === 'Path' && path.ref.type === 'Free' && isKeyword(path.ref.name)) {\n      let { name } = path.ref;\n\n      let usedType = this.#type;\n      let validTypes = KEYWORDS_TYPES[name];\n\n      if (validTypes.indexOf(usedType) === -1) {\n        return Err(\n          generateSyntaxError(\n            `The \\`${name}\\` keyword was used incorrectly. It was used as ${\n              typesToReadableName[usedType]\n            }, but its valid usages are:\\n\\n${generateTypesMessage(\n              name,\n              validTypes\n            )}\\n\\nError caused by`,\n            node.loc\n          )\n        );\n      }\n    }\n\n    return null;\n  }\n}\n\nconst typesToReadableName = {\n  Append: 'an append statement',\n  Block: 'a block statement',\n  Call: 'a call expression',\n  Modifier: 'a modifier',\n};\n\nfunction generateTypesMessage(name: string, types: KeywordType[]): string {\n  return types\n    .map((type) => {\n      switch (type) {\n        case 'Append':\n          return `- As an append statement, as in: {{${name}}}`;\n        case 'Block':\n          return `- As a block statement, as in: {{#${name}}}{{/${name}}}`;\n        case 'Call':\n          return `- As an expression, as in: (${name})`;\n        case 'Modifier':\n          return `- As a modifier, as in: <div {{${name}}}></div>`;\n        default:\n          return exhausted(type);\n      }\n    })\n    .join('\\n\\n');\n}\n\n/**\n * This function builds keyword definitions for a particular type of AST node (`KeywordType`).\n *\n * You can build keyword definitions for:\n *\n * - `Expr`: A `SubExpression` or `PathExpression`\n * - `Block`: A `BlockStatement`\n *   - A `BlockStatement` is a keyword candidate if its head is a\n *     `PathExpression`\n * - `Append`: An `AppendStatement`\n *\n * A node is a keyword candidate if:\n *\n * - A `PathExpression` is a keyword candidate if it has no tail, and its\n *   head expression is a `LocalVarHead` or `FreeVarHead` whose name is\n *   the keyword's name.\n * - A `SubExpression`, `AppendStatement`, or `BlockStatement` is a keyword\n *   candidate if its head is a keyword candidate.\n *\n * The keyword infrastructure guarantees that:\n *\n * - If a node is not a keyword candidate, it is never passed to any keyword's\n *   `assert` method.\n * - If a node is not the `KeywordType` for a particular keyword, it will not\n *   be passed to the keyword's `assert` method.\n *\n * `Expr` keywords are used in expression positions and should return HIR\n * expressions. `Block` and `Append` keywords are used in statement\n * positions and should return HIR statements.\n *\n * A keyword definition has two parts:\n *\n * - `match`, which determines whether an AST node matches the keyword, and can\n *   optionally return some information extracted from the AST node.\n * - `translate`, which takes a matching AST node as well as the extracted\n *   information and returns an appropriate HIR instruction.\n *\n * # Example\n *\n * This keyword:\n *\n * - turns `(hello)` into `\"hello\"`\n *   - as long as `hello` is not in scope\n * - makes it an error to pass any arguments (such as `(hello world)`)\n *\n * ```ts\n * keywords('SubExpr').kw('hello', {\n *   assert(node: ExprKeywordNode): Result<void> | false {\n *     // we don't want to transform `hello` as a `PathExpression`\n *     if (node.type !== 'SubExpression') {\n *       return false;\n *     }\n *\n *     // node.head would be `LocalVarHead` if `hello` was in scope\n *     if (node.head.type !== 'FreeVarHead') {\n *       return false;\n *     }\n *\n *     if (node.params.length || node.hash) {\n *       return Err(generateSyntaxError(`(hello) does not take any arguments`), node.loc);\n *     } else {\n *       return Ok();\n *     }\n *   },\n *\n *   translate(node: ASTv2.SubExpression): hir.Expression {\n *     return ASTv2.builders.literal(\"hello\", node.loc)\n *   }\n * })\n * ```\n *\n * The keyword infrastructure checks to make sure that the node is the right\n * type before calling `assert`, so you only need to consider `SubExpression`\n * and `PathExpression` here. It also checks to make sure that the node passed\n * to `assert` has the keyword name in the right place.\n *\n * Note the important difference between returning `false` from `assert`,\n * which just means that the node didn't match, and returning `Err`, which\n * means that the node matched, but there was a keyword-specific syntax\n * error.\n */\nexport function keywords<K extends KeywordType>(type: K): Keywords<K> {\n  return new Keywords(type);\n}\n"],"sourceRoot":""} |
@@ -1,2 +0,2 @@ | ||
export { precompile, precompileJSON, PrecompileOptions } from './lib/compiler'; | ||
export { defaultId, precompile, precompileJSON, PrecompileOptions } from './lib/compiler'; | ||
export { ProgramSymbols, buildStatement, buildStatements, s, c, unicode, NEWLINE, } from './lib/builder/builder'; | ||
@@ -3,0 +3,0 @@ export { BuilderStatement, Builder } from './lib/builder/builder-interface'; |
import { SerializedTemplateBlock, TemplateJavascript } from '@glimmer/interfaces'; | ||
import { PrecompileOptions } from '@glimmer/syntax'; | ||
import { PrecompileOptions, TemplateIdFn } from '@glimmer/syntax'; | ||
export declare const defaultId: TemplateIdFn; | ||
export declare function precompileJSON(string: string, options?: PrecompileOptions): [block: SerializedTemplateBlock, usedLocals: string[]]; | ||
@@ -4,0 +5,0 @@ export declare function precompile(source: string, options?: PrecompileOptions): TemplateJavascript; |
{ | ||
"name": "@glimmer/compiler", | ||
"version": "0.76.0", | ||
"version": "0.77.0", | ||
"repository": "https://github.com/glimmerjs/glimmer-vm/tree/master/packages/@glimmer/compiler", | ||
"dependencies": { | ||
"@glimmer/syntax": "0.76.0", | ||
"@glimmer/util": "0.76.0", | ||
"@glimmer/wire-format": "0.76.0", | ||
"@glimmer/interfaces": "0.76.0", | ||
"@glimmer/syntax": "0.77.0", | ||
"@glimmer/util": "0.77.0", | ||
"@glimmer/wire-format": "0.77.0", | ||
"@glimmer/interfaces": "0.77.0", | ||
"@simple-dom/interface": "^1.4.0" | ||
}, | ||
"devDependencies": { | ||
"@glimmer/local-debug-flags": "0.76.0" | ||
"@glimmer/local-debug-flags": "0.77.0" | ||
}, | ||
@@ -15,0 +15,0 @@ "main": "dist/commonjs/es2017/index.js", |
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
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
2858503
25969
6
+ Added@glimmer/interfaces@0.77.0(transitive)
+ Added@glimmer/syntax@0.77.0(transitive)
+ Added@glimmer/util@0.77.0(transitive)
+ Added@glimmer/wire-format@0.77.0(transitive)
- Removed@glimmer/interfaces@0.76.0(transitive)
- Removed@glimmer/syntax@0.76.0(transitive)
- Removed@glimmer/util@0.76.0(transitive)
- Removed@glimmer/wire-format@0.76.0(transitive)
Updated@glimmer/interfaces@0.77.0
Updated@glimmer/syntax@0.77.0
Updated@glimmer/util@0.77.0
Updated@glimmer/wire-format@0.77.0