dev.cel.common.CelProtoAbstractSyntaxTree Maven / Gradle / Ivy
Show all versions of validators Show documentation
// Copyright 2022 Google LLC
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// https://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package dev.cel.common;
import static com.google.common.base.Preconditions.checkState;
import static com.google.common.collect.ImmutableList.toImmutableList;
import static com.google.common.collect.ImmutableMap.toImmutableMap;
import dev.cel.expr.CheckedExpr;
import dev.cel.expr.Expr;
import dev.cel.expr.ParsedExpr;
import dev.cel.expr.SourceInfo;
import dev.cel.expr.SourceInfo.Extension;
import dev.cel.expr.SourceInfo.Extension.Component;
import dev.cel.expr.SourceInfo.Extension.Version;
import dev.cel.expr.Type;
import com.google.common.collect.ImmutableList;
import com.google.errorprone.annotations.CheckReturnValue;
import dev.cel.common.ast.CelExprConverter;
import dev.cel.common.types.CelTypes;
import java.util.Collection;
import java.util.Map.Entry;
/**
* An Adapter for {@link CelAbstractSyntaxTree} constructed from Canonical Protos of {@link
* CheckedExpr} or {@link ParsedExpr} expressions.
*
* Note: Keep this file in sync with {@link CelProtoV1Alpha1AbstractSyntaxTree}
*/
// LINT.IfChange
public final class CelProtoAbstractSyntaxTree {
private final CheckedExpr checkedExpr;
private final CelAbstractSyntaxTree ast;
private CelProtoAbstractSyntaxTree(CheckedExpr checkedExpr) {
this.checkedExpr = checkedExpr;
this.ast =
CelAbstractSyntaxTree.newCheckedAst(
CelExprConverter.fromExpr(checkedExpr.getExpr()),
CelSource.newBuilder()
.addAllLineOffsets(checkedExpr.getSourceInfo().getLineOffsetsList())
.addPositionsMap(checkedExpr.getSourceInfo().getPositionsMap())
.addAllMacroCalls(
CelExprConverter.exprMacroCallsToCelExprMacroCalls(
checkedExpr.getSourceInfo().getMacroCallsMap()))
.addAllExtensions(
fromExprExtensionsToCelExtensions(
checkedExpr.getSourceInfo().getExtensionsList()))
.setDescription(checkedExpr.getSourceInfo().getLocation())
.build(),
checkedExpr.getReferenceMapMap().entrySet().stream()
.collect(
toImmutableMap(
Entry::getKey,
v -> CelExprConverter.exprReferenceToCelReference(v.getValue()))),
checkedExpr.getTypeMapMap().entrySet().stream()
.collect(toImmutableMap(Entry::getKey, v -> CelTypes.typeToCelType(v.getValue()))));
}
private CelProtoAbstractSyntaxTree(CelAbstractSyntaxTree ast) {
this.ast = ast;
CheckedExpr.Builder checkedExprBuilder =
CheckedExpr.newBuilder()
.setSourceInfo(
SourceInfo.newBuilder()
.setLocation(ast.getSource().getDescription())
.addAllLineOffsets(ast.getSource().getLineOffsets())
.addAllExtensions(
fromCelExtensionsToExprExtensions(ast.getSource().getExtensions()))
.putAllMacroCalls(
ast.getSource().getMacroCalls().entrySet().stream()
.collect(
toImmutableMap(
Entry::getKey,
v -> CelExprConverter.fromCelExpr(v.getValue()))))
.putAllPositions(ast.getSource().getPositionsMap()))
.setExpr(CelExprConverter.fromCelExpr(ast.getExpr()));
if (ast.isChecked()) {
checkedExprBuilder.putAllReferenceMap(
ast.getReferenceMap().entrySet().stream()
.collect(
toImmutableMap(
Entry::getKey,
v -> CelExprConverter.celReferenceToExprReference(v.getValue()))));
checkedExprBuilder.putAllTypeMap(
ast.getTypeMap().entrySet().stream()
.collect(toImmutableMap(Entry::getKey, v -> CelTypes.celTypeToType(v.getValue()))));
}
this.checkedExpr = checkedExprBuilder.build();
}
/** Construct an abstract syntax tree from a {@link com.google.api.expr.CheckedExpr}. */
public static CelProtoAbstractSyntaxTree fromCheckedExpr(CheckedExpr checkedExpr) {
return new CelProtoAbstractSyntaxTree(checkedExpr);
}
/** Construct an abstract syntax tree from a {@link com.google.api.expr.ParsedExpr}. */
public static CelProtoAbstractSyntaxTree fromParsedExpr(ParsedExpr parsedExpr) {
return new CelProtoAbstractSyntaxTree(
CheckedExpr.newBuilder()
.setExpr(parsedExpr.getExpr())
.setSourceInfo(parsedExpr.getSourceInfo())
.build());
}
/** Constructs CelProtoAbstractSyntaxTree from {@link CelAbstractSyntaxTree}. */
public static CelProtoAbstractSyntaxTree fromCelAst(CelAbstractSyntaxTree ast) {
return new CelProtoAbstractSyntaxTree(ast);
}
/** Tests whether the underlying abstract syntax tree has been type checked or not. */
public boolean isChecked() {
return ast.isChecked();
}
/**
* Returns the native representation of the abstract syntax tree: {@link CelAbstractSyntaxTree}.
*/
@CheckReturnValue
public CelAbstractSyntaxTree getAst() {
return ast;
}
/**
* Returns the underlying {@link com.google.api.expr.Expr} representation of the abstract syntax
* tree.
*/
@CheckReturnValue
public Expr getExpr() {
return checkedExpr.getExpr();
}
/**
* Returns the underlying {@link com.google.api.expr.CheckedExpr} representation of the abstract
* syntax tree. Throws {@link java.lang.IllegalStateException} if {@link
* CelAbstractSyntaxTree#isChecked} is false.
*/
@CheckReturnValue
public CheckedExpr toCheckedExpr() {
checkState(
isChecked(),
"CelAbstractSyntaxTree must be checked before it can be converted to CheckedExpr");
return checkedExpr;
}
/**
* Returns the underlying {@link com.google.api.expr.SourceInfo} representation of the abstract
* syntax tree.
*/
@CheckReturnValue
public SourceInfo getSourceInfo() {
return checkedExpr.getSourceInfo();
}
/**
* Returns the underlying {@link com.google.api.expr.ParsedExpr} representation of the abstract
* syntax tree.
*/
@CheckReturnValue
public ParsedExpr toParsedExpr() {
return ParsedExpr.newBuilder().setExpr(getExpr()).setSourceInfo(getSourceInfo()).build();
}
/**
* For a type checked abstract syntax tree the resulting type is returned in proto format
* described in checked.proto. Otherwise, the dynamic type is returned.
*/
@CheckReturnValue
public Type getProtoResultType() {
return CelTypes.celTypeToType(ast.getResultType());
}
private static ImmutableList fromCelExtensionsToExprExtensions(
Collection extensions) {
return extensions.stream()
.map(
celSourceExtension ->
Extension.newBuilder()
.setId(celSourceExtension.id())
.setVersion(
Version.newBuilder()
.setMajor(celSourceExtension.version().major())
.setMinor(celSourceExtension.version().minor()))
.addAllAffectedComponents(
celSourceExtension.affectedComponents().stream()
.map(
component -> {
switch (component) {
case COMPONENT_UNSPECIFIED:
return Component.COMPONENT_UNSPECIFIED;
case COMPONENT_PARSER:
return Component.COMPONENT_PARSER;
case COMPONENT_TYPE_CHECKER:
return Component.COMPONENT_TYPE_CHECKER;
case COMPONENT_RUNTIME:
return Component.COMPONENT_RUNTIME;
}
throw new AssertionError(
"Unexpected component kind: " + component);
})
.collect(toImmutableList()))
.build())
.collect(toImmutableList());
}
private static ImmutableList fromExprExtensionsToCelExtensions(
Collection extensions) {
return extensions.stream()
.map(
exprExtension ->
CelSource.Extension.create(
exprExtension.getId(),
CelSource.Extension.Version.of(
exprExtension.getVersion().getMajor(),
exprExtension.getVersion().getMinor()),
exprExtension.getAffectedComponentsList().stream()
.map(
component -> {
switch (component) {
case COMPONENT_UNSPECIFIED:
return CelSource.Extension.Component.COMPONENT_UNSPECIFIED;
case COMPONENT_PARSER:
return CelSource.Extension.Component.COMPONENT_PARSER;
case COMPONENT_TYPE_CHECKER:
return CelSource.Extension.Component.COMPONENT_TYPE_CHECKER;
case COMPONENT_RUNTIME:
return CelSource.Extension.Component.COMPONENT_RUNTIME;
case UNRECOGNIZED:
// fall-through
}
throw new AssertionError("Unexpected component kind: " + component);
})
.collect(toImmutableList())))
.collect(toImmutableList());
}
}
// LINT.ThenChange(CelProtoV1Alpha1AbstractSyntaxTree.java)