@glimmer/compiler
Advanced tools
Comparing version 0.73.0 to 0.73.1
@@ -56,2 +56,10 @@ "use strict"; | ||
if (path !== null && path.ref.type === 'Free') { | ||
if (path.tail.length > 0) { | ||
if (path.ref.resolution.serialize() === 'Loose') { | ||
// cannot be a keyword reference, keywords do not allow paths (must be | ||
// relying on implicit this fallback) | ||
return false; | ||
} | ||
} | ||
return path.ref.name === this.keyword; | ||
@@ -65,2 +73,8 @@ } else { | ||
if (this.match(node)) { | ||
let path = getPathExpression(node); | ||
if (path !== null && 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)); | ||
} | ||
let param = this.delegate.assert(node, state); | ||
@@ -270,2 +284,2 @@ return param.andThen(param => this.delegate.translate({ | ||
} | ||
//# 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      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 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 = 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":""} |
@@ -103,2 +103,10 @@ "use strict"; | ||
if (path !== null && path.ref.type === 'Free') { | ||
if (path.tail.length > 0) { | ||
if (path.ref.resolution.serialize() === 'Loose') { | ||
// cannot be a keyword reference, keywords do not allow paths (must be | ||
// relying on implicit this fallback) | ||
return false; | ||
} | ||
} | ||
return path.ref.name === this.keyword; | ||
@@ -114,2 +122,8 @@ } else { | ||
if (this.match(node)) { | ||
var path = getPathExpression(node); | ||
if (path !== null && 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)); | ||
} | ||
var param = this.delegate.assert(node, state); | ||
@@ -325,2 +339,2 @@ return param.andThen(function (param) { | ||
} | ||
//# 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      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 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":";;;;;;;;;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":""} |
@@ -45,2 +45,10 @@ var __classPrivateFieldSet = this && this.__classPrivateFieldSet || function (receiver, privateMap, value) { | ||
if (path !== null && path.ref.type === 'Free') { | ||
if (path.tail.length > 0) { | ||
if (path.ref.resolution.serialize() === 'Loose') { | ||
// cannot be a keyword reference, keywords do not allow paths (must be | ||
// relying on implicit this fallback) | ||
return false; | ||
} | ||
} | ||
return path.ref.name === this.keyword; | ||
@@ -54,2 +62,8 @@ } else { | ||
if (this.match(node)) { | ||
let path = getPathExpression(node); | ||
if (path !== null && 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)); | ||
} | ||
let param = this.delegate.assert(node, state); | ||
@@ -255,2 +269,2 @@ return param.andThen(param => this.delegate.translate({ | ||
} | ||
//# 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      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 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 = 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":""} |
@@ -54,2 +54,10 @@ 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); } | ||
if (path !== null && path.ref.type === 'Free') { | ||
if (path.tail.length > 0) { | ||
if (path.ref.resolution.serialize() === 'Loose') { | ||
// cannot be a keyword reference, keywords do not allow paths (must be | ||
// relying on implicit this fallback) | ||
return false; | ||
} | ||
} | ||
return path.ref.name === this.keyword; | ||
@@ -65,2 +73,8 @@ } else { | ||
if (this.match(node)) { | ||
var path = getPathExpression(node); | ||
if (path !== null && 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)); | ||
} | ||
var param = this.delegate.assert(node, state); | ||
@@ -272,2 +286,2 @@ return param.andThen(function (param) { | ||
} | ||
//# 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      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 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 = 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":""} |
{ | ||
"name": "@glimmer/compiler", | ||
"version": "0.73.0", | ||
"version": "0.73.1", | ||
"repository": "https://github.com/glimmerjs/glimmer-vm/tree/master/packages/@glimmer/compiler", | ||
"dependencies": { | ||
"@glimmer/syntax": "0.73.0", | ||
"@glimmer/util": "0.73.0", | ||
"@glimmer/wire-format": "0.73.0", | ||
"@glimmer/interfaces": "0.73.0", | ||
"@glimmer/syntax": "0.73.1", | ||
"@glimmer/util": "0.73.1", | ||
"@glimmer/wire-format": "0.73.1", | ||
"@glimmer/interfaces": "0.73.1", | ||
"@simple-dom/interface": "^1.4.0" | ||
}, | ||
"devDependencies": { | ||
"@glimmer/local-debug-flags": "0.73.0" | ||
"@glimmer/local-debug-flags": "0.73.1" | ||
}, | ||
@@ -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
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
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
2854324
25931
+ Added@glimmer/interfaces@0.73.1(transitive)
+ Added@glimmer/syntax@0.73.1(transitive)
+ Added@glimmer/util@0.73.1(transitive)
+ Added@glimmer/wire-format@0.73.1(transitive)
- Removed@glimmer/interfaces@0.73.0(transitive)
- Removed@glimmer/syntax@0.73.0(transitive)
- Removed@glimmer/util@0.73.0(transitive)
- Removed@glimmer/wire-format@0.73.0(transitive)
Updated@glimmer/interfaces@0.73.1
Updated@glimmer/syntax@0.73.1
Updated@glimmer/util@0.73.1
Updated@glimmer/wire-format@0.73.1