@angular/compiler 16.0.0-next.7 → 16.0.0-rc.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/src/render3/partial/class_metadata.mjs +1 -1
- package/esm2022/src/render3/partial/directive.mjs +1 -1
- package/esm2022/src/render3/partial/factory.mjs +1 -1
- package/esm2022/src/render3/partial/injectable.mjs +1 -1
- package/esm2022/src/render3/partial/injector.mjs +1 -1
- package/esm2022/src/render3/partial/ng_module.mjs +1 -1
- package/esm2022/src/render3/partial/pipe.mjs +1 -1
- package/esm2022/src/render3/view/compiler.mjs +52 -27
- package/esm2022/src/render3/view/util.mjs +4 -2
- package/esm2022/src/template/pipeline/ir/index.mjs +17 -0
- package/esm2022/src/template/pipeline/ir/src/element.mjs +101 -0
- package/esm2022/src/template/pipeline/ir/src/enums.mjs +121 -0
- package/esm2022/src/template/pipeline/ir/src/expression.mjs +293 -0
- package/esm2022/src/template/pipeline/ir/src/operations.mjs +223 -0
- package/esm2022/src/template/pipeline/ir/src/ops/create.mjs +78 -0
- package/esm2022/src/template/pipeline/ir/src/ops/shared.mjs +43 -0
- package/esm2022/src/template/pipeline/ir/src/ops/update.mjs +49 -0
- package/esm2022/src/template/pipeline/ir/src/traits.mjs +68 -0
- package/esm2022/src/template/pipeline/ir/src/variable.mjs +9 -0
- package/esm2022/src/template/pipeline/src/compilation.mjs +122 -0
- package/esm2022/src/template/pipeline/src/emit.mjs +83 -0
- package/esm2022/src/template/pipeline/src/ingest.mjs +194 -0
- package/esm2022/src/template/pipeline/src/instruction.mjs +139 -0
- package/esm2022/src/template/pipeline/src/phases/const_collection.mjs +57 -0
- package/esm2022/src/template/pipeline/src/phases/empty_elements.mjs +27 -0
- package/esm2022/src/template/pipeline/src/phases/generate_advance.mjs +56 -0
- package/esm2022/src/template/pipeline/src/phases/generate_variables.mjs +146 -0
- package/esm2022/src/template/pipeline/src/phases/local_refs.mjs +44 -0
- package/esm2022/src/template/pipeline/src/phases/naming.mjs +61 -0
- package/esm2022/src/template/pipeline/src/phases/reify.mjs +157 -0
- package/esm2022/src/template/pipeline/src/phases/resolve_contexts.mjs +55 -0
- package/esm2022/src/template/pipeline/src/phases/resolve_names.mjs +95 -0
- package/esm2022/src/template/pipeline/src/phases/slot_allocation.mjs +75 -0
- package/esm2022/src/template/pipeline/src/phases/var_counting.mjs +60 -0
- package/esm2022/src/template/pipeline/switch/index.mjs +2 -0
- package/esm2022/src/version.mjs +1 -1
- package/fesm2022/compiler.mjs +2386 -231
- package/fesm2022/compiler.mjs.map +1 -1
- package/fesm2022/testing.mjs +1 -1
- package/index.d.ts +1 -1
- package/package.json +3 -3
- package/testing/index.d.ts +1 -1
|
@@ -0,0 +1,293 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright Google LLC All Rights Reserved.
|
|
4
|
+
*
|
|
5
|
+
* Use of this source code is governed by an MIT-style license that can be
|
|
6
|
+
* found in the LICENSE file at https://angular.io/license
|
|
7
|
+
*/
|
|
8
|
+
var _a;
|
|
9
|
+
import * as o from '../../../../output/output_ast';
|
|
10
|
+
import { ExpressionKind, OpKind } from './enums';
|
|
11
|
+
import { UsesSlotIndex } from './traits';
|
|
12
|
+
/**
|
|
13
|
+
* Check whether a given `o.Expression` is a logical IR expression type.
|
|
14
|
+
*/
|
|
15
|
+
export function isIrExpression(expr) {
|
|
16
|
+
return expr instanceof ExpressionBase;
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* Base type used for all logical IR expressions.
|
|
20
|
+
*/
|
|
21
|
+
export class ExpressionBase extends o.Expression {
|
|
22
|
+
constructor(sourceSpan = null) {
|
|
23
|
+
super(null, sourceSpan);
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* Logical expression representing a lexical read of a variable name.
|
|
28
|
+
*/
|
|
29
|
+
export class LexicalReadExpr extends ExpressionBase {
|
|
30
|
+
constructor(name) {
|
|
31
|
+
super();
|
|
32
|
+
this.name = name;
|
|
33
|
+
this.kind = ExpressionKind.LexicalRead;
|
|
34
|
+
}
|
|
35
|
+
visitExpression(visitor, context) { }
|
|
36
|
+
isEquivalent() {
|
|
37
|
+
return false;
|
|
38
|
+
}
|
|
39
|
+
isConstant() {
|
|
40
|
+
return false;
|
|
41
|
+
}
|
|
42
|
+
transformInternalExpressions() { }
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Runtime operation to retrieve the value of a local reference.
|
|
46
|
+
*/
|
|
47
|
+
export class ReferenceExpr extends ExpressionBase {
|
|
48
|
+
static { _a = UsesSlotIndex; }
|
|
49
|
+
constructor(target, offset) {
|
|
50
|
+
super();
|
|
51
|
+
this.target = target;
|
|
52
|
+
this.offset = offset;
|
|
53
|
+
this.kind = ExpressionKind.Reference;
|
|
54
|
+
this[_a] = true;
|
|
55
|
+
this.slot = null;
|
|
56
|
+
}
|
|
57
|
+
visitExpression() { }
|
|
58
|
+
isEquivalent(e) {
|
|
59
|
+
return e instanceof ReferenceExpr && e.target === this.target;
|
|
60
|
+
}
|
|
61
|
+
isConstant() {
|
|
62
|
+
return false;
|
|
63
|
+
}
|
|
64
|
+
transformInternalExpressions() { }
|
|
65
|
+
}
|
|
66
|
+
/**
|
|
67
|
+
* A reference to the current view context (usually the `ctx` variable in a template function).
|
|
68
|
+
*/
|
|
69
|
+
export class ContextExpr extends ExpressionBase {
|
|
70
|
+
constructor(view) {
|
|
71
|
+
super();
|
|
72
|
+
this.view = view;
|
|
73
|
+
this.kind = ExpressionKind.Context;
|
|
74
|
+
}
|
|
75
|
+
visitExpression() { }
|
|
76
|
+
isEquivalent(e) {
|
|
77
|
+
return e instanceof ContextExpr && e.view === this.view;
|
|
78
|
+
}
|
|
79
|
+
isConstant() {
|
|
80
|
+
return false;
|
|
81
|
+
}
|
|
82
|
+
transformInternalExpressions() { }
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* Runtime operation to navigate to the next view context in the view hierarchy.
|
|
86
|
+
*/
|
|
87
|
+
export class NextContextExpr extends ExpressionBase {
|
|
88
|
+
constructor() {
|
|
89
|
+
super();
|
|
90
|
+
this.kind = ExpressionKind.NextContext;
|
|
91
|
+
}
|
|
92
|
+
visitExpression() { }
|
|
93
|
+
isEquivalent(e) {
|
|
94
|
+
return e instanceof NextContextExpr;
|
|
95
|
+
}
|
|
96
|
+
isConstant() {
|
|
97
|
+
return false;
|
|
98
|
+
}
|
|
99
|
+
transformInternalExpressions() { }
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Runtime operation to snapshot the current view context.
|
|
103
|
+
*
|
|
104
|
+
* The result of this operation can be stored in a variable and later used with the `RestoreView`
|
|
105
|
+
* operation.
|
|
106
|
+
*/
|
|
107
|
+
export class GetCurrentViewExpr extends ExpressionBase {
|
|
108
|
+
constructor() {
|
|
109
|
+
super();
|
|
110
|
+
this.kind = ExpressionKind.GetCurrentView;
|
|
111
|
+
}
|
|
112
|
+
visitExpression() { }
|
|
113
|
+
isEquivalent(e) {
|
|
114
|
+
return e instanceof GetCurrentViewExpr;
|
|
115
|
+
}
|
|
116
|
+
isConstant() {
|
|
117
|
+
return false;
|
|
118
|
+
}
|
|
119
|
+
transformInternalExpressions() { }
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* Runtime operation to restore a snapshotted view.
|
|
123
|
+
*/
|
|
124
|
+
export class RestoreViewExpr extends ExpressionBase {
|
|
125
|
+
constructor(view) {
|
|
126
|
+
super();
|
|
127
|
+
this.view = view;
|
|
128
|
+
this.kind = ExpressionKind.RestoreView;
|
|
129
|
+
}
|
|
130
|
+
visitExpression(visitor, context) {
|
|
131
|
+
if (typeof this.view !== 'number') {
|
|
132
|
+
this.view.visitExpression(visitor, context);
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
isEquivalent(e) {
|
|
136
|
+
if (!(e instanceof RestoreViewExpr) || typeof e.view !== typeof this.view) {
|
|
137
|
+
return false;
|
|
138
|
+
}
|
|
139
|
+
if (typeof this.view === 'number') {
|
|
140
|
+
return this.view === e.view;
|
|
141
|
+
}
|
|
142
|
+
else {
|
|
143
|
+
return this.view.isEquivalent(e.view);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
isConstant() {
|
|
147
|
+
return false;
|
|
148
|
+
}
|
|
149
|
+
transformInternalExpressions(transform) {
|
|
150
|
+
if (typeof this.view !== 'number') {
|
|
151
|
+
this.view = transformExpressionsInExpression(this.view, transform);
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* Runtime operation to reset the current view context after `RestoreView`.
|
|
157
|
+
*/
|
|
158
|
+
export class ResetViewExpr extends ExpressionBase {
|
|
159
|
+
constructor(expr) {
|
|
160
|
+
super();
|
|
161
|
+
this.expr = expr;
|
|
162
|
+
this.kind = ExpressionKind.ResetView;
|
|
163
|
+
}
|
|
164
|
+
visitExpression(visitor, context) {
|
|
165
|
+
this.expr.visitExpression(visitor, context);
|
|
166
|
+
}
|
|
167
|
+
isEquivalent(e) {
|
|
168
|
+
return e instanceof ResetViewExpr && this.expr.isEquivalent(e.expr);
|
|
169
|
+
}
|
|
170
|
+
isConstant() {
|
|
171
|
+
return false;
|
|
172
|
+
}
|
|
173
|
+
transformInternalExpressions(transform) {
|
|
174
|
+
this.expr = transformExpressionsInExpression(this.expr, transform);
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
/**
|
|
178
|
+
* Read of a variable declared as an `ir.VariableOp` and referenced through its `ir.XrefId`.
|
|
179
|
+
*/
|
|
180
|
+
export class ReadVariableExpr extends ExpressionBase {
|
|
181
|
+
constructor(xref) {
|
|
182
|
+
super();
|
|
183
|
+
this.xref = xref;
|
|
184
|
+
this.kind = ExpressionKind.ReadVariable;
|
|
185
|
+
this.name = null;
|
|
186
|
+
}
|
|
187
|
+
visitExpression() { }
|
|
188
|
+
isEquivalent(other) {
|
|
189
|
+
return other instanceof ReadVariableExpr && other.xref === this.xref;
|
|
190
|
+
}
|
|
191
|
+
isConstant() {
|
|
192
|
+
return false;
|
|
193
|
+
}
|
|
194
|
+
transformInternalExpressions() { }
|
|
195
|
+
}
|
|
196
|
+
/**
|
|
197
|
+
* Visits all `Expression`s in the AST of `op` with the `visitor` function.
|
|
198
|
+
*/
|
|
199
|
+
export function visitExpressionsInOp(op, visitor) {
|
|
200
|
+
transformExpressionsInOp(op, (expr) => {
|
|
201
|
+
visitor(expr);
|
|
202
|
+
return expr;
|
|
203
|
+
});
|
|
204
|
+
}
|
|
205
|
+
/**
|
|
206
|
+
* Transform all `Expression`s in the AST of `op` with the `transform` function.
|
|
207
|
+
*
|
|
208
|
+
* All such operations will be replaced with the result of applying `transform`, which may be an
|
|
209
|
+
* identity transformation.
|
|
210
|
+
*/
|
|
211
|
+
export function transformExpressionsInOp(op, transform) {
|
|
212
|
+
switch (op.kind) {
|
|
213
|
+
case OpKind.Property:
|
|
214
|
+
op.expression = transformExpressionsInExpression(op.expression, transform);
|
|
215
|
+
break;
|
|
216
|
+
case OpKind.Statement:
|
|
217
|
+
transformExpressionsInStatement(op.statement, transform);
|
|
218
|
+
break;
|
|
219
|
+
case OpKind.Variable:
|
|
220
|
+
op.initializer = transformExpressionsInExpression(op.initializer, transform);
|
|
221
|
+
break;
|
|
222
|
+
case OpKind.InterpolateText:
|
|
223
|
+
for (let i = 0; i < op.expressions.length; i++) {
|
|
224
|
+
op.expressions[i] = transformExpressionsInExpression(op.expressions[i], transform);
|
|
225
|
+
}
|
|
226
|
+
break;
|
|
227
|
+
case OpKind.Listener:
|
|
228
|
+
for (const innerOp of op.handlerOps) {
|
|
229
|
+
transformExpressionsInOp(innerOp, transform);
|
|
230
|
+
}
|
|
231
|
+
break;
|
|
232
|
+
case OpKind.Element:
|
|
233
|
+
case OpKind.ElementStart:
|
|
234
|
+
case OpKind.ElementEnd:
|
|
235
|
+
case OpKind.Template:
|
|
236
|
+
case OpKind.Text:
|
|
237
|
+
// These operations contain no expressions.
|
|
238
|
+
break;
|
|
239
|
+
default:
|
|
240
|
+
throw new Error(`AssertionError: transformExpressionsInOp doesn't handle ${OpKind[op.kind]}`);
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
/**
|
|
244
|
+
* Transform all `Expression`s in the AST of `expr` with the `transform` function.
|
|
245
|
+
*
|
|
246
|
+
* All such operations will be replaced with the result of applying `transform`, which may be an
|
|
247
|
+
* identity transformation.
|
|
248
|
+
*/
|
|
249
|
+
export function transformExpressionsInExpression(expr, transform) {
|
|
250
|
+
if (expr instanceof ExpressionBase) {
|
|
251
|
+
expr.transformInternalExpressions(transform);
|
|
252
|
+
return transform(expr);
|
|
253
|
+
}
|
|
254
|
+
else if (expr instanceof o.BinaryOperatorExpr) {
|
|
255
|
+
expr.lhs = transformExpressionsInExpression(expr.lhs, transform);
|
|
256
|
+
expr.rhs = transformExpressionsInExpression(expr.rhs, transform);
|
|
257
|
+
}
|
|
258
|
+
else if (expr instanceof o.ReadPropExpr) {
|
|
259
|
+
expr.receiver = transformExpressionsInExpression(expr.receiver, transform);
|
|
260
|
+
}
|
|
261
|
+
else if (expr instanceof o.InvokeFunctionExpr) {
|
|
262
|
+
expr.fn = transformExpressionsInExpression(expr.fn, transform);
|
|
263
|
+
for (let i = 0; i < expr.args.length; i++) {
|
|
264
|
+
expr.args[i] = transformExpressionsInExpression(expr.args[i], transform);
|
|
265
|
+
}
|
|
266
|
+
}
|
|
267
|
+
else if (expr instanceof o.ReadVarExpr || expr instanceof o.ExternalExpr ||
|
|
268
|
+
expr instanceof o.LiteralExpr) {
|
|
269
|
+
// No action for these types.
|
|
270
|
+
}
|
|
271
|
+
else {
|
|
272
|
+
throw new Error(`Unhandled expression kind: ${expr.constructor.name}`);
|
|
273
|
+
}
|
|
274
|
+
return expr;
|
|
275
|
+
}
|
|
276
|
+
/**
|
|
277
|
+
* Transform all `Expression`s in the AST of `stmt` with the `transform` function.
|
|
278
|
+
*
|
|
279
|
+
* All such operations will be replaced with the result of applying `transform`, which may be an
|
|
280
|
+
* identity transformation.
|
|
281
|
+
*/
|
|
282
|
+
export function transformExpressionsInStatement(stmt, transform) {
|
|
283
|
+
if (stmt instanceof o.ExpressionStatement) {
|
|
284
|
+
stmt.expr = transformExpressionsInExpression(stmt.expr, transform);
|
|
285
|
+
}
|
|
286
|
+
else if (stmt instanceof o.ReturnStatement) {
|
|
287
|
+
stmt.value = transformExpressionsInExpression(stmt.value, transform);
|
|
288
|
+
}
|
|
289
|
+
else {
|
|
290
|
+
throw new Error(`Unhandled statement kind: ${stmt.constructor.name}`);
|
|
291
|
+
}
|
|
292
|
+
}
|
|
293
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"expression.js","sourceRoot":"","sources":["../../../../../../../../../../packages/compiler/src/template/pipeline/ir/src/expression.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport * as o from '../../../../output/output_ast';\nimport type {ParseSourceSpan} from '../../../../parse_util';\n\nimport {ExpressionKind, OpKind} from './enums';\nimport {UsesSlotIndex, UsesSlotIndexExprTrait} from './traits';\n\nimport type {XrefId} from './operations';\nimport type {CreateOp} from './ops/create';\nimport type {UpdateOp} from './ops/update';\n\n/**\n * An `o.Expression` subtype representing a logical expression in the intermediate representation.\n */\nexport type Expression = LexicalReadExpr|ReferenceExpr|ContextExpr|NextContextExpr|\n    GetCurrentViewExpr|RestoreViewExpr|ResetViewExpr|ReadVariableExpr;\n\n/**\n * Transformer type which converts IR expressions into general `o.Expression`s (which may be an\n * identity transformation).\n */\nexport type ExpressionTransform = (expr: Expression) => o.Expression;\n\n/**\n * Check whether a given `o.Expression` is a logical IR expression type.\n */\nexport function isIrExpression(expr: o.Expression): boolean {\n  return expr instanceof ExpressionBase;\n}\n\n/**\n * Base type used for all logical IR expressions.\n */\nexport abstract class ExpressionBase extends o.Expression {\n  abstract readonly kind: ExpressionKind;\n\n  constructor(sourceSpan: ParseSourceSpan|null = null) {\n    super(null, sourceSpan);\n  }\n\n  /**\n   * Run the transformer against any nested expressions which may be present in this IR expression\n   * subtype.\n   */\n  abstract transformInternalExpressions(transform: ExpressionTransform): void;\n}\n\n/**\n * Logical expression representing a lexical read of a variable name.\n */\nexport class LexicalReadExpr extends ExpressionBase {\n  override readonly kind = ExpressionKind.LexicalRead;\n\n  constructor(readonly name: string) {\n    super();\n  }\n\n  override visitExpression(visitor: o.ExpressionVisitor, context: any): void {}\n\n  override isEquivalent(): boolean {\n    return false;\n  }\n\n  override isConstant(): boolean {\n    return false;\n  }\n\n  override transformInternalExpressions(): void {}\n}\n\n/**\n * Runtime operation to retrieve the value of a local reference.\n */\nexport class ReferenceExpr extends ExpressionBase implements UsesSlotIndexExprTrait {\n  override readonly kind = ExpressionKind.Reference;\n\n  readonly[UsesSlotIndex] = true;\n\n  slot: number|null = null;\n\n  constructor(readonly target: XrefId, readonly offset: number) {\n    super();\n  }\n\n  override visitExpression(): void {}\n\n  override isEquivalent(e: o.Expression): boolean {\n    return e instanceof ReferenceExpr && e.target === this.target;\n  }\n\n  override isConstant(): boolean {\n    return false;\n  }\n\n  override transformInternalExpressions(): void {}\n}\n\n/**\n * A reference to the current view context (usually the `ctx` variable in a template function).\n */\nexport class ContextExpr extends ExpressionBase {\n  override readonly kind = ExpressionKind.Context;\n\n  constructor(readonly view: XrefId) {\n    super();\n  }\n\n  override visitExpression(): void {}\n\n  override isEquivalent(e: o.Expression): boolean {\n    return e instanceof ContextExpr && e.view === this.view;\n  }\n\n  override isConstant(): boolean {\n    return false;\n  }\n\n  override transformInternalExpressions(): void {}\n}\n\n/**\n * Runtime operation to navigate to the next view context in the view hierarchy.\n */\nexport class NextContextExpr extends ExpressionBase {\n  override readonly kind = ExpressionKind.NextContext;\n\n  constructor() {\n    super();\n  }\n\n  override visitExpression(): void {}\n\n  override isEquivalent(e: o.Expression): boolean {\n    return e instanceof NextContextExpr;\n  }\n\n  override isConstant(): boolean {\n    return false;\n  }\n\n  override transformInternalExpressions(): void {}\n}\n\n/**\n * Runtime operation to snapshot the current view context.\n *\n * The result of this operation can be stored in a variable and later used with the `RestoreView`\n * operation.\n */\nexport class GetCurrentViewExpr extends ExpressionBase {\n  override readonly kind = ExpressionKind.GetCurrentView;\n\n  constructor() {\n    super();\n  }\n\n  override visitExpression(): void {}\n\n  override isEquivalent(e: o.Expression): boolean {\n    return e instanceof GetCurrentViewExpr;\n  }\n\n  override isConstant(): boolean {\n    return false;\n  }\n\n  override transformInternalExpressions(): void {}\n}\n\n/**\n * Runtime operation to restore a snapshotted view.\n */\nexport class RestoreViewExpr extends ExpressionBase {\n  override readonly kind = ExpressionKind.RestoreView;\n\n  constructor(public view: XrefId|o.Expression) {\n    super();\n  }\n\n  override visitExpression(visitor: o.ExpressionVisitor, context: any): void {\n    if (typeof this.view !== 'number') {\n      this.view.visitExpression(visitor, context);\n    }\n  }\n\n  override isEquivalent(e: o.Expression): boolean {\n    if (!(e instanceof RestoreViewExpr) || typeof e.view !== typeof this.view) {\n      return false;\n    }\n\n    if (typeof this.view === 'number') {\n      return this.view === e.view;\n    } else {\n      return this.view.isEquivalent(e.view as o.Expression);\n    }\n  }\n\n  override isConstant(): boolean {\n    return false;\n  }\n\n  override transformInternalExpressions(transform: ExpressionTransform): void {\n    if (typeof this.view !== 'number') {\n      this.view = transformExpressionsInExpression(this.view, transform);\n    }\n  }\n}\n\n/**\n * Runtime operation to reset the current view context after `RestoreView`.\n */\nexport class ResetViewExpr extends ExpressionBase {\n  override readonly kind = ExpressionKind.ResetView;\n\n  constructor(public expr: o.Expression) {\n    super();\n  }\n\n  override visitExpression(visitor: o.ExpressionVisitor, context: any): any {\n    this.expr.visitExpression(visitor, context);\n  }\n\n  override isEquivalent(e: o.Expression): boolean {\n    return e instanceof ResetViewExpr && this.expr.isEquivalent(e.expr);\n  }\n\n  override isConstant(): boolean {\n    return false;\n  }\n\n  override transformInternalExpressions(transform: ExpressionTransform): void {\n    this.expr = transformExpressionsInExpression(this.expr, transform);\n  }\n}\n\n/**\n * Read of a variable declared as an `ir.VariableOp` and referenced through its `ir.XrefId`.\n */\nexport class ReadVariableExpr extends ExpressionBase {\n  override readonly kind = ExpressionKind.ReadVariable;\n  name: string|null = null;\n  constructor(readonly xref: XrefId) {\n    super();\n  }\n\n  override visitExpression(): void {}\n\n  override isEquivalent(other: o.Expression): boolean {\n    return other instanceof ReadVariableExpr && other.xref === this.xref;\n  }\n\n  override isConstant(): boolean {\n    return false;\n  }\n\n  override transformInternalExpressions(): void {}\n}\n\n/**\n * Visits all `Expression`s in the AST of `op` with the `visitor` function.\n */\nexport function visitExpressionsInOp(\n    op: CreateOp|UpdateOp, visitor: (expr: Expression) => void): void {\n  transformExpressionsInOp(op, (expr) => {\n    visitor(expr);\n    return expr;\n  });\n}\n\n/**\n * Transform all `Expression`s in the AST of `op` with the `transform` function.\n *\n * All such operations will be replaced with the result of applying `transform`, which may be an\n * identity transformation.\n */\nexport function transformExpressionsInOp(\n    op: CreateOp|UpdateOp, transform: ExpressionTransform): void {\n  switch (op.kind) {\n    case OpKind.Property:\n      op.expression = transformExpressionsInExpression(op.expression, transform);\n      break;\n    case OpKind.Statement:\n      transformExpressionsInStatement(op.statement, transform);\n      break;\n    case OpKind.Variable:\n      op.initializer = transformExpressionsInExpression(op.initializer, transform);\n      break;\n    case OpKind.InterpolateText:\n      for (let i = 0; i < op.expressions.length; i++) {\n        op.expressions[i] = transformExpressionsInExpression(op.expressions[i], transform);\n      }\n      break;\n    case OpKind.Listener:\n      for (const innerOp of op.handlerOps) {\n        transformExpressionsInOp(innerOp, transform);\n      }\n      break;\n    case OpKind.Element:\n    case OpKind.ElementStart:\n    case OpKind.ElementEnd:\n    case OpKind.Template:\n    case OpKind.Text:\n      // These operations contain no expressions.\n      break;\n    default:\n      throw new Error(`AssertionError: transformExpressionsInOp doesn't handle ${OpKind[op.kind]}`);\n  }\n}\n\n/**\n * Transform all `Expression`s in the AST of `expr` with the `transform` function.\n *\n * All such operations will be replaced with the result of applying `transform`, which may be an\n * identity transformation.\n */\nexport function transformExpressionsInExpression(\n    expr: o.Expression, transform: ExpressionTransform): o.Expression {\n  if (expr instanceof ExpressionBase) {\n    expr.transformInternalExpressions(transform);\n    return transform(expr as Expression);\n  } else if (expr instanceof o.BinaryOperatorExpr) {\n    expr.lhs = transformExpressionsInExpression(expr.lhs, transform);\n    expr.rhs = transformExpressionsInExpression(expr.rhs, transform);\n  } else if (expr instanceof o.ReadPropExpr) {\n    expr.receiver = transformExpressionsInExpression(expr.receiver, transform);\n  } else if (expr instanceof o.InvokeFunctionExpr) {\n    expr.fn = transformExpressionsInExpression(expr.fn, transform);\n    for (let i = 0; i < expr.args.length; i++) {\n      expr.args[i] = transformExpressionsInExpression(expr.args[i], transform);\n    }\n  } else if (\n      expr instanceof o.ReadVarExpr || expr instanceof o.ExternalExpr ||\n      expr instanceof o.LiteralExpr) {\n    // No action for these types.\n  } else {\n    throw new Error(`Unhandled expression kind: ${expr.constructor.name}`);\n  }\n  return expr;\n}\n\n/**\n * Transform all `Expression`s in the AST of `stmt` with the `transform` function.\n *\n * All such operations will be replaced with the result of applying `transform`, which may be an\n * identity transformation.\n */\nexport function transformExpressionsInStatement(\n    stmt: o.Statement, transform: ExpressionTransform): void {\n  if (stmt instanceof o.ExpressionStatement) {\n    stmt.expr = transformExpressionsInExpression(stmt.expr, transform);\n  } else if (stmt instanceof o.ReturnStatement) {\n    stmt.value = transformExpressionsInExpression(stmt.value, transform);\n  } else {\n    throw new Error(`Unhandled statement kind: ${stmt.constructor.name}`);\n  }\n}\n"]}
|
|
@@ -0,0 +1,223 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright Google LLC All Rights Reserved.
|
|
4
|
+
*
|
|
5
|
+
* Use of this source code is governed by an MIT-style license that can be
|
|
6
|
+
* found in the LICENSE file at https://angular.io/license
|
|
7
|
+
*/
|
|
8
|
+
import { OpKind } from './enums';
|
|
9
|
+
/**
|
|
10
|
+
* A linked list of `Op` nodes of a given subtype.
|
|
11
|
+
*
|
|
12
|
+
* @param OpT specific subtype of `Op` nodes which this list contains.
|
|
13
|
+
*/
|
|
14
|
+
class OpList {
|
|
15
|
+
static { this.nextListId = 0; }
|
|
16
|
+
constructor() {
|
|
17
|
+
/**
|
|
18
|
+
* Debug ID of this `OpList` instance.
|
|
19
|
+
*/
|
|
20
|
+
this.debugListId = OpList.nextListId++;
|
|
21
|
+
// OpList uses static head/tail nodes of a special `ListEnd` type.
|
|
22
|
+
// This avoids the need for special casing of the first and last list
|
|
23
|
+
// elements in all list operations.
|
|
24
|
+
this.head = {
|
|
25
|
+
kind: OpKind.ListEnd,
|
|
26
|
+
next: null,
|
|
27
|
+
prev: null,
|
|
28
|
+
debugListId: this.debugListId,
|
|
29
|
+
};
|
|
30
|
+
this.tail = {
|
|
31
|
+
kind: OpKind.ListEnd,
|
|
32
|
+
next: null,
|
|
33
|
+
prev: null,
|
|
34
|
+
debugListId: this.debugListId,
|
|
35
|
+
};
|
|
36
|
+
// Link `head` and `tail` together at the start (list is empty).
|
|
37
|
+
this.head.next = this.tail;
|
|
38
|
+
this.tail.prev = this.head;
|
|
39
|
+
}
|
|
40
|
+
/**
|
|
41
|
+
* Push a new operation to the tail of the list.
|
|
42
|
+
*/
|
|
43
|
+
push(op) {
|
|
44
|
+
OpList.assertIsNotEnd(op);
|
|
45
|
+
OpList.assertIsUnowned(op);
|
|
46
|
+
op.debugListId = this.debugListId;
|
|
47
|
+
// The old "previous" node (which might be the head, if the list is empty).
|
|
48
|
+
const oldLast = this.tail.prev;
|
|
49
|
+
// Insert `op` following the old last node.
|
|
50
|
+
op.prev = oldLast;
|
|
51
|
+
oldLast.next = op;
|
|
52
|
+
// Connect `op` with the list tail.
|
|
53
|
+
op.next = this.tail;
|
|
54
|
+
this.tail.prev = op;
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Prepend one or more nodes to the start of the list.
|
|
58
|
+
*/
|
|
59
|
+
prepend(ops) {
|
|
60
|
+
if (ops.length === 0) {
|
|
61
|
+
return;
|
|
62
|
+
}
|
|
63
|
+
for (const op of ops) {
|
|
64
|
+
OpList.assertIsNotEnd(op);
|
|
65
|
+
OpList.assertIsUnowned(op);
|
|
66
|
+
op.debugListId = this.debugListId;
|
|
67
|
+
}
|
|
68
|
+
const first = this.head.next;
|
|
69
|
+
let prev = this.head;
|
|
70
|
+
for (const op of ops) {
|
|
71
|
+
prev.next = op;
|
|
72
|
+
op.prev = prev;
|
|
73
|
+
prev = op;
|
|
74
|
+
}
|
|
75
|
+
prev.next = first;
|
|
76
|
+
first.prev = prev;
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* `OpList` is iterable via the iteration protocol.
|
|
80
|
+
*
|
|
81
|
+
* It's safe to mutate the part of the list that has already been returned by the iterator, up to
|
|
82
|
+
* and including the last operation returned. Mutations beyond that point _may_ be safe, but may
|
|
83
|
+
* also corrupt the iteration position and should be avoided.
|
|
84
|
+
*/
|
|
85
|
+
*[Symbol.iterator]() {
|
|
86
|
+
let current = this.head.next;
|
|
87
|
+
while (current !== this.tail) {
|
|
88
|
+
// Guards against corruption of the iterator state by mutations to the tail of the list during
|
|
89
|
+
// iteration.
|
|
90
|
+
OpList.assertIsOwned(current);
|
|
91
|
+
const next = current.next;
|
|
92
|
+
yield current;
|
|
93
|
+
current = next;
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
/**
|
|
97
|
+
* Replace `oldOp` with `newOp` in the list.
|
|
98
|
+
*/
|
|
99
|
+
static replace(oldOp, newOp) {
|
|
100
|
+
OpList.assertIsNotEnd(oldOp);
|
|
101
|
+
OpList.assertIsNotEnd(newOp);
|
|
102
|
+
OpList.assertIsOwned(oldOp);
|
|
103
|
+
OpList.assertIsUnowned(newOp);
|
|
104
|
+
newOp.debugListId = oldOp.debugListId;
|
|
105
|
+
if (oldOp.prev !== null) {
|
|
106
|
+
oldOp.prev.next = newOp;
|
|
107
|
+
newOp.prev = oldOp.prev;
|
|
108
|
+
}
|
|
109
|
+
if (oldOp.next !== null) {
|
|
110
|
+
oldOp.next.prev = newOp;
|
|
111
|
+
newOp.next = oldOp.next;
|
|
112
|
+
}
|
|
113
|
+
oldOp.debugListId = null;
|
|
114
|
+
oldOp.prev = null;
|
|
115
|
+
oldOp.next = null;
|
|
116
|
+
}
|
|
117
|
+
/**
|
|
118
|
+
* Replace `oldOp` with some number of new operations in the list (which may include `oldOp`).
|
|
119
|
+
*/
|
|
120
|
+
static replaceWithMany(oldOp, newOps) {
|
|
121
|
+
if (newOps.length === 0) {
|
|
122
|
+
// Replacing with an empty list -> pure removal.
|
|
123
|
+
OpList.remove(oldOp);
|
|
124
|
+
return;
|
|
125
|
+
}
|
|
126
|
+
OpList.assertIsNotEnd(oldOp);
|
|
127
|
+
OpList.assertIsOwned(oldOp);
|
|
128
|
+
const listId = oldOp.debugListId;
|
|
129
|
+
oldOp.debugListId = null;
|
|
130
|
+
for (const newOp of newOps) {
|
|
131
|
+
OpList.assertIsNotEnd(newOp);
|
|
132
|
+
// `newOp` might be `oldOp`, but at this point it's been marked as unowned.
|
|
133
|
+
OpList.assertIsUnowned(newOp);
|
|
134
|
+
}
|
|
135
|
+
// It should be safe to reuse `oldOp` in the `newOps` list - maybe you want to sandwich an
|
|
136
|
+
// operation between two new ops.
|
|
137
|
+
const { prev: oldPrev, next: oldNext } = oldOp;
|
|
138
|
+
oldOp.prev = null;
|
|
139
|
+
oldOp.next = null;
|
|
140
|
+
let prev = oldPrev;
|
|
141
|
+
for (const newOp of newOps) {
|
|
142
|
+
this.assertIsUnowned(newOp);
|
|
143
|
+
newOp.debugListId = listId;
|
|
144
|
+
prev.next = newOp;
|
|
145
|
+
newOp.prev = prev;
|
|
146
|
+
// This _should_ be the case, but set it just in case.
|
|
147
|
+
newOp.next = null;
|
|
148
|
+
prev = newOp;
|
|
149
|
+
}
|
|
150
|
+
// At the end of iteration, `prev` holds the last node in the list.
|
|
151
|
+
const first = newOps[0];
|
|
152
|
+
const last = prev;
|
|
153
|
+
// Replace `oldOp` with the chain `first` -> `last`.
|
|
154
|
+
if (oldPrev !== null) {
|
|
155
|
+
oldPrev.next = first;
|
|
156
|
+
first.prev = oldOp.prev;
|
|
157
|
+
}
|
|
158
|
+
if (oldNext !== null) {
|
|
159
|
+
oldNext.prev = last;
|
|
160
|
+
last.next = oldNext;
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
/**
|
|
164
|
+
* Remove the given node from the list which contains it.
|
|
165
|
+
*/
|
|
166
|
+
static remove(op) {
|
|
167
|
+
OpList.assertIsNotEnd(op);
|
|
168
|
+
OpList.assertIsOwned(op);
|
|
169
|
+
op.prev.next = op.next;
|
|
170
|
+
op.next.prev = op.prev;
|
|
171
|
+
// Break any link between the node and this list to safeguard against its usage in future
|
|
172
|
+
// operations.
|
|
173
|
+
op.debugListId = null;
|
|
174
|
+
op.prev = null;
|
|
175
|
+
op.next = null;
|
|
176
|
+
}
|
|
177
|
+
/**
|
|
178
|
+
* Insert `op` before `before`.
|
|
179
|
+
*/
|
|
180
|
+
static insertBefore(op, before) {
|
|
181
|
+
OpList.assertIsNotEnd(before);
|
|
182
|
+
OpList.assertIsNotEnd(op);
|
|
183
|
+
OpList.assertIsUnowned(op);
|
|
184
|
+
OpList.assertIsOwned(before, op.debugListId);
|
|
185
|
+
op.debugListId = before.debugListId;
|
|
186
|
+
// Just in case.
|
|
187
|
+
op.prev = null;
|
|
188
|
+
before.prev.next = op;
|
|
189
|
+
op.prev = before.prev;
|
|
190
|
+
op.next = before;
|
|
191
|
+
before.prev = op;
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* Asserts that `op` does not currently belong to a list.
|
|
195
|
+
*/
|
|
196
|
+
static assertIsUnowned(op) {
|
|
197
|
+
if (op.debugListId !== null) {
|
|
198
|
+
throw new Error(`AssertionError: illegal operation on owned node: ${OpKind[op.kind]}`);
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
/**
|
|
202
|
+
* Asserts that `op` currently belongs to a list. If `byList` is passed, `op` is asserted to
|
|
203
|
+
* specifically belong to that list.
|
|
204
|
+
*/
|
|
205
|
+
static assertIsOwned(op, byList) {
|
|
206
|
+
if (op.debugListId === null) {
|
|
207
|
+
throw new Error(`AssertionError: illegal operation on unowned node: ${OpKind[op.kind]}`);
|
|
208
|
+
}
|
|
209
|
+
else if (byList !== undefined && op.debugListId !== byList) {
|
|
210
|
+
throw new Error(`AssertionError: node belongs to the wrong list (expected ${byList}, actual ${op.debugListId})`);
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
/**
|
|
214
|
+
* Asserts that `op` is not a special `ListEnd` node.
|
|
215
|
+
*/
|
|
216
|
+
static assertIsNotEnd(op) {
|
|
217
|
+
if (op.kind === OpKind.ListEnd) {
|
|
218
|
+
throw new Error(`AssertionError: illegal operation on list head or tail`);
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
export { OpList };
|
|
223
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"operations.js","sourceRoot":"","sources":["../../../../../../../../../../packages/compiler/src/template/pipeline/ir/src/operations.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {OpKind} from './enums';\n\n/**\n * Branded type for a cross-reference ID. During ingest, `XrefId`s are generated to link together\n * different IR operations which need to reference each other.\n */\nexport type XrefId = number&{__brand: 'XrefId'};\n\n/**\n * Base interface for semantic operations being performed within a template.\n *\n * @param OpT a specific narrower type of `Op` (for example, creation operations) which this\n *     specific subtype of `Op` can be linked with in a linked list.\n */\nexport interface Op<OpT extends Op<OpT>> {\n  /**\n   * All operations have a distinct kind.\n   */\n  kind: OpKind;\n\n  /**\n   * The previous operation in the linked list, if any.\n   *\n   * This is `null` for operation nodes not currently in a list, or for the special head/tail nodes.\n   */\n  prev: OpT|null;\n\n  /**\n   * The next operation in the linked list, if any.\n   *\n   * This is `null` for operation nodes not currently in a list, or for the special head/tail nodes.\n   */\n  next: OpT|null;\n\n  /**\n   * Debug id of the list to which this node currently belongs, or `null` if this node is not part\n   * of a list.\n   */\n  debugListId: number|null;\n}\n\n/**\n * A linked list of `Op` nodes of a given subtype.\n *\n * @param OpT specific subtype of `Op` nodes which this list contains.\n */\nexport class OpList<OpT extends Op<OpT>> {\n  static nextListId = 0;\n\n  /**\n   * Debug ID of this `OpList` instance.\n   */\n  readonly debugListId = OpList.nextListId++;\n\n  // OpList uses static head/tail nodes of a special `ListEnd` type.\n  // This avoids the need for special casing of the first and last list\n  // elements in all list operations.\n  readonly head: OpT = {\n    kind: OpKind.ListEnd,\n    next: null,\n    prev: null,\n    debugListId: this.debugListId,\n  } as OpT;\n\n  readonly tail = {\n    kind: OpKind.ListEnd,\n    next: null,\n    prev: null,\n    debugListId: this.debugListId,\n  } as OpT;\n\n\n  constructor() {\n    // Link `head` and `tail` together at the start (list is empty).\n    this.head.next = this.tail;\n    this.tail.prev = this.head;\n  }\n\n  /**\n   * Push a new operation to the tail of the list.\n   */\n  push(op: OpT): void {\n    OpList.assertIsNotEnd(op);\n    OpList.assertIsUnowned(op);\n\n    op.debugListId = this.debugListId;\n\n    // The old \"previous\" node (which might be the head, if the list is empty).\n    const oldLast = this.tail.prev!;\n\n    // Insert `op` following the old last node.\n    op.prev = oldLast;\n    oldLast.next = op;\n\n    // Connect `op` with the list tail.\n    op.next = this.tail;\n    this.tail.prev = op;\n  }\n\n  /**\n   * Prepend one or more nodes to the start of the list.\n   */\n  prepend(ops: OpT[]): void {\n    if (ops.length === 0) {\n      return;\n    }\n\n    for (const op of ops) {\n      OpList.assertIsNotEnd(op);\n      OpList.assertIsUnowned(op);\n\n      op.debugListId = this.debugListId;\n    }\n\n    const first = this.head.next!;\n\n    let prev = this.head;\n    for (const op of ops) {\n      prev.next = op;\n      op.prev = prev;\n\n      prev = op;\n    }\n\n    prev.next = first;\n    first.prev = prev;\n  }\n\n  /**\n   * `OpList` is iterable via the iteration protocol.\n   *\n   * It's safe to mutate the part of the list that has already been returned by the iterator, up to\n   * and including the last operation returned. Mutations beyond that point _may_ be safe, but may\n   * also corrupt the iteration position and should be avoided.\n   */\n  * [Symbol.iterator](): Generator<OpT> {\n    let current = this.head.next!;\n    while (current !== this.tail) {\n      // Guards against corruption of the iterator state by mutations to the tail of the list during\n      // iteration.\n      OpList.assertIsOwned(current);\n\n      const next = current.next!;\n      yield current;\n      current = next;\n    }\n  }\n\n  /**\n   * Replace `oldOp` with `newOp` in the list.\n   */\n  static replace<OpT extends Op<OpT>>(oldOp: OpT, newOp: OpT): void {\n    OpList.assertIsNotEnd(oldOp);\n    OpList.assertIsNotEnd(newOp);\n\n    OpList.assertIsOwned(oldOp);\n    OpList.assertIsUnowned(newOp);\n\n    newOp.debugListId = oldOp.debugListId;\n    if (oldOp.prev !== null) {\n      oldOp.prev.next = newOp;\n      newOp.prev = oldOp.prev;\n    }\n    if (oldOp.next !== null) {\n      oldOp.next.prev = newOp;\n      newOp.next = oldOp.next;\n    }\n    oldOp.debugListId = null;\n    oldOp.prev = null;\n    oldOp.next = null;\n  }\n\n  /**\n   * Replace `oldOp` with some number of new operations in the list (which may include `oldOp`).\n   */\n  static replaceWithMany<OpT extends Op<OpT>>(oldOp: OpT, newOps: OpT[]): void {\n    if (newOps.length === 0) {\n      // Replacing with an empty list -> pure removal.\n      OpList.remove(oldOp);\n      return;\n    }\n\n    OpList.assertIsNotEnd(oldOp);\n    OpList.assertIsOwned(oldOp);\n\n    const listId = oldOp.debugListId;\n    oldOp.debugListId = null;\n\n    for (const newOp of newOps) {\n      OpList.assertIsNotEnd(newOp);\n\n      // `newOp` might be `oldOp`, but at this point it's been marked as unowned.\n      OpList.assertIsUnowned(newOp);\n    }\n\n    // It should be safe to reuse `oldOp` in the `newOps` list - maybe you want to sandwich an\n    // operation between two new ops.\n    const {prev: oldPrev, next: oldNext} = oldOp;\n    oldOp.prev = null;\n    oldOp.next = null;\n\n    let prev: OpT = oldPrev!;\n    for (const newOp of newOps) {\n      this.assertIsUnowned(newOp);\n      newOp.debugListId = listId;\n\n      prev!.next = newOp;\n      newOp.prev = prev;\n\n      // This _should_ be the case, but set it just in case.\n      newOp.next = null;\n\n      prev = newOp;\n    }\n    // At the end of iteration, `prev` holds the last node in the list.\n    const first = newOps[0]!;\n    const last = prev!;\n\n    // Replace `oldOp` with the chain `first` -> `last`.\n    if (oldPrev !== null) {\n      oldPrev.next = first;\n      first.prev = oldOp.prev;\n    }\n\n    if (oldNext !== null) {\n      oldNext.prev = last;\n      last.next = oldNext;\n    }\n  }\n\n  /**\n   * Remove the given node from the list which contains it.\n   */\n  static remove<OpT extends Op<OpT>>(op: OpT): void {\n    OpList.assertIsNotEnd(op);\n    OpList.assertIsOwned(op);\n\n    op.prev!.next = op.next;\n    op.next!.prev = op.prev;\n\n    // Break any link between the node and this list to safeguard against its usage in future\n    // operations.\n    op.debugListId = null;\n    op.prev = null;\n    op.next = null;\n  }\n\n  /**\n   * Insert `op` before `before`.\n   */\n  static insertBefore<OpT extends Op<OpT>>(op: OpT, before: OpT): void {\n    OpList.assertIsNotEnd(before);\n    OpList.assertIsNotEnd(op);\n\n    OpList.assertIsUnowned(op);\n    OpList.assertIsOwned(before, op.debugListId!);\n\n    op.debugListId = before.debugListId;\n\n    // Just in case.\n    op.prev = null;\n\n    before.prev!.next = op;\n    op.prev = before.prev;\n\n    op.next = before;\n    before.prev = op;\n  }\n\n  /**\n   * Asserts that `op` does not currently belong to a list.\n   */\n  static assertIsUnowned<OpT extends Op<OpT>>(op: OpT): void {\n    if (op.debugListId !== null) {\n      throw new Error(`AssertionError: illegal operation on owned node: ${OpKind[op.kind]}`);\n    }\n  }\n\n  /**\n   * Asserts that `op` currently belongs to a list. If `byList` is passed, `op` is asserted to\n   * specifically belong to that list.\n   */\n  static assertIsOwned<OpT extends Op<OpT>>(op: OpT, byList?: number): void {\n    if (op.debugListId === null) {\n      throw new Error(`AssertionError: illegal operation on unowned node: ${OpKind[op.kind]}`);\n    } else if (byList !== undefined && op.debugListId !== byList) {\n      throw new Error(`AssertionError: node belongs to the wrong list (expected ${byList}, actual ${\n          op.debugListId})`);\n    }\n  }\n\n  /**\n   * Asserts that `op` is not a special `ListEnd` node.\n   */\n  static assertIsNotEnd<OpT extends Op<OpT>>(op: OpT): void {\n    if (op.kind === OpKind.ListEnd) {\n      throw new Error(`AssertionError: illegal operation on list head or tail`);\n    }\n  }\n}\n"]}
|