dev.cel.common.ast.CelExprConverter 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.ast;
import static com.google.common.collect.ImmutableMap.toImmutableMap;
import dev.cel.expr.Constant;
import dev.cel.expr.Expr;
import dev.cel.expr.Expr.Call;
import dev.cel.expr.Expr.Comprehension;
import dev.cel.expr.Expr.CreateList;
import dev.cel.expr.Expr.CreateStruct;
import dev.cel.expr.Expr.CreateStruct.Entry;
import dev.cel.expr.Expr.CreateStruct.Entry.KeyKindCase;
import dev.cel.expr.Expr.Ident;
import dev.cel.expr.Expr.Select;
import dev.cel.expr.Reference;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.primitives.UnsignedLong;
import java.util.Map;
import java.util.Optional;
/**
* Converts a proto-based Canonical Expression Tree {@link Expr} into the CEL native representation
* of Expression tree {@link CelExpr} and vice versa.
*
* Note: Keep this file in sync with {@code CelExprV1Alpha1Converter}
*/
// LINT.IfChange
public final class CelExprConverter {
private CelExprConverter() {}
/** Convert {@link CelExpr} into {@link Expr} */
public static Expr fromCelExpr(CelExpr celExpr) {
CelExpr.ExprKind celExprKind = celExpr.exprKind();
Expr.Builder expr = newExprBuilder(celExpr);
switch (celExprKind.getKind()) {
case CONSTANT:
return expr.setConstExpr(celConstantToExprConstant(celExprKind.constant())).build();
case IDENT:
return expr.setIdentExpr(Ident.newBuilder().setName(celExprKind.ident().name())).build();
case SELECT:
CelExpr.CelSelect celSelect = celExprKind.select();
return expr.setSelectExpr(
Select.newBuilder()
.setField(celSelect.field())
.setOperand(fromCelExpr(celSelect.operand()))
.setTestOnly(celSelect.testOnly()))
.build();
case CALL:
CelExpr.CelCall celCall = celExprKind.call();
Call.Builder callBuilder =
Call.newBuilder()
.setFunction(celCall.function())
.addAllArgs(fromCelExprList(celCall.args()));
celCall.target().ifPresent(target -> callBuilder.setTarget(fromCelExpr(target)));
return expr.setCallExpr(callBuilder).build();
case CREATE_LIST:
CelExpr.CelCreateList celCreateList = celExprKind.createList();
return expr.setListExpr(
CreateList.newBuilder()
.addAllElements(fromCelExprList(celCreateList.elements()))
.addAllOptionalIndices(celCreateList.optionalIndices()))
.build();
case CREATE_STRUCT:
return expr.setStructExpr(celStructToExprStruct(celExprKind.createStruct())).build();
case CREATE_MAP:
return expr.setStructExpr(celMapToExprStruct(celExprKind.createMap())).build();
case COMPREHENSION:
CelExpr.CelComprehension celComprehension = celExprKind.comprehension();
return expr.setComprehensionExpr(
Comprehension.newBuilder()
.setIterVar(celComprehension.iterVar())
.setIterRange(fromCelExpr(celComprehension.iterRange()))
.setAccuVar(celComprehension.accuVar())
.setAccuInit(fromCelExpr(celComprehension.accuInit()))
.setLoopCondition(fromCelExpr(celComprehension.loopCondition()))
.setLoopStep(fromCelExpr(celComprehension.loopStep()))
.setResult(fromCelExpr(celComprehension.result())))
.build();
case NOT_SET:
return expr.build();
}
throw new IllegalArgumentException(
"Unexpected expression kind case: " + celExpr.exprKind().getKind());
}
/** Convert {@link Expr} into {@link CelExpr} */
public static CelExpr fromExpr(Expr expr) {
switch (expr.getExprKindCase()) {
case CONST_EXPR:
return CelExpr.ofConstantExpr(expr.getId(), exprConstantToCelConstant(expr.getConstExpr()));
case IDENT_EXPR:
return CelExpr.ofIdentExpr(expr.getId(), expr.getIdentExpr().getName());
case SELECT_EXPR:
Select selectExpr = expr.getSelectExpr();
return CelExpr.ofSelectExpr(
expr.getId(),
fromExpr(selectExpr.getOperand()),
selectExpr.getField(),
selectExpr.getTestOnly());
case CALL_EXPR:
Call callExpr = expr.getCallExpr();
return CelExpr.ofCallExpr(
expr.getId(),
callExpr.hasTarget() ? Optional.of(fromExpr(callExpr.getTarget())) : Optional.empty(),
callExpr.getFunction(),
fromExprList(callExpr.getArgsList()));
case LIST_EXPR:
CreateList createListExpr = expr.getListExpr();
return CelExpr.ofCreateListExpr(
expr.getId(),
fromExprList(createListExpr.getElementsList()),
ImmutableList.copyOf(createListExpr.getOptionalIndicesList()));
case STRUCT_EXPR:
return exprStructToCelStruct(expr.getId(), expr.getStructExpr());
case COMPREHENSION_EXPR:
Comprehension comprehensionExpr = expr.getComprehensionExpr();
return CelExpr.ofComprehension(
expr.getId(),
comprehensionExpr.getIterVar(),
fromExpr(comprehensionExpr.getIterRange()),
comprehensionExpr.getAccuVar(),
fromExpr(comprehensionExpr.getAccuInit()),
fromExpr(comprehensionExpr.getLoopCondition()),
fromExpr(comprehensionExpr.getLoopStep()),
fromExpr(comprehensionExpr.getResult()));
case EXPRKIND_NOT_SET:
return CelExpr.ofNotSet(expr.getId());
}
throw new IllegalArgumentException(
"Unexpected expression kind case: " + expr.getExprKindCase());
}
private static ImmutableList fromExprList(Iterable exprList) {
ImmutableList.Builder celExprListBuilder = ImmutableList.builder();
for (Expr expr : exprList) {
celExprListBuilder.add(fromExpr(expr));
}
return celExprListBuilder.build();
}
/**
* Converts a proto-based {@link Reference} to CEL native representation of {@link CelReference}.
*/
public static CelReference exprReferenceToCelReference(Reference reference) {
CelReference.Builder builder =
CelReference.newBuilder()
.setName(reference.getName())
.addOverloadIds(reference.getOverloadIdList());
if (reference.hasValue()) {
builder.setValue(exprConstantToCelConstant(reference.getValue()));
}
return builder.build();
}
/**
* Converts a proto-based {@link Constant} to CEL native representation of {@link CelConstant}.
*/
public static CelConstant exprConstantToCelConstant(Constant constExpr) {
switch (constExpr.getConstantKindCase()) {
case CONSTANTKIND_NOT_SET:
return CelConstant.ofNotSet();
case NULL_VALUE:
return CelConstant.ofValue(constExpr.getNullValue());
case BOOL_VALUE:
return CelConstant.ofValue(constExpr.getBoolValue());
case INT64_VALUE:
return CelConstant.ofValue(constExpr.getInt64Value());
case UINT64_VALUE:
return CelConstant.ofValue(UnsignedLong.fromLongBits(constExpr.getUint64Value()));
case DOUBLE_VALUE:
return CelConstant.ofValue(constExpr.getDoubleValue());
case STRING_VALUE:
return CelConstant.ofValue(constExpr.getStringValue());
case BYTES_VALUE:
return CelConstant.ofValue(constExpr.getBytesValue());
case DURATION_VALUE:
return CelConstant.ofValue(constExpr.getDurationValue());
case TIMESTAMP_VALUE:
return CelConstant.ofValue(constExpr.getTimestampValue());
default:
throw new IllegalStateException(
"unsupported constant case: " + constExpr.getConstantKindCase());
}
}
private static CelExpr exprStructToCelStruct(long id, CreateStruct structExpr) {
if (!structExpr.getMessageName().isEmpty()) {
ImmutableList.Builder entries = ImmutableList.builder();
for (Entry structExprEntry : structExpr.getEntriesList()) {
if (!structExprEntry.getKeyKindCase().equals(KeyKindCase.FIELD_KEY)) {
throw new IllegalArgumentException(
"Unexpected struct key kind case: " + structExprEntry.getKeyKindCase());
}
entries.add(
CelExpr.ofCreateStructEntryExpr(
structExprEntry.getId(),
structExprEntry.getFieldKey(),
fromExpr(structExprEntry.getValue()),
structExprEntry.getOptionalEntry()));
}
return CelExpr.ofCreateStructExpr(id, structExpr.getMessageName(), entries.build());
} else {
ImmutableList.Builder entries = ImmutableList.builder();
for (Entry mapExprEntry : structExpr.getEntriesList()) {
if (!mapExprEntry.getKeyKindCase().equals(KeyKindCase.MAP_KEY)) {
throw new IllegalArgumentException(
"Unexpected map key kind case: " + mapExprEntry.getKeyKindCase());
}
entries.add(
CelExpr.ofCreateMapEntryExpr(
mapExprEntry.getId(),
fromExpr(mapExprEntry.getMapKey()),
fromExpr(mapExprEntry.getValue()),
mapExprEntry.getOptionalEntry()));
}
return CelExpr.ofCreateMapExpr(id, entries.build());
}
}
private static Expr.Builder newExprBuilder(CelExpr expr) {
return Expr.newBuilder().setId(expr.id());
}
/**
* Converts a proto-based {@link Constant} to CEL native representation of {@link CelConstant}.
*/
public static Constant celConstantToExprConstant(CelConstant celConstant) {
switch (celConstant.getKind()) {
case NOT_SET:
return Constant.getDefaultInstance();
case NULL_VALUE:
return Constant.newBuilder().setNullValue(celConstant.nullValue()).build();
case BOOLEAN_VALUE:
return Constant.newBuilder().setBoolValue(celConstant.booleanValue()).build();
case INT64_VALUE:
return Constant.newBuilder().setInt64Value(celConstant.int64Value()).build();
case UINT64_VALUE:
return Constant.newBuilder().setUint64Value(celConstant.uint64Value().longValue()).build();
case DOUBLE_VALUE:
return Constant.newBuilder().setDoubleValue(celConstant.doubleValue()).build();
case STRING_VALUE:
return Constant.newBuilder().setStringValue(celConstant.stringValue()).build();
case BYTES_VALUE:
return Constant.newBuilder().setBytesValue(celConstant.bytesValue()).build();
case DURATION_VALUE:
return Constant.newBuilder().setDurationValue(celConstant.durationValue()).build();
case TIMESTAMP_VALUE:
return Constant.newBuilder().setTimestampValue(celConstant.timestampValue()).build();
}
throw new IllegalStateException("unsupported constant case: " + celConstant.getKind());
}
private static CreateStruct celStructToExprStruct(CelExpr.CelCreateStruct celCreateStruct) {
ImmutableList.Builder entries = ImmutableList.builder();
for (CelExpr.CelCreateStruct.Entry celStructExprEntry : celCreateStruct.entries()) {
entries.add(
CreateStruct.Entry.newBuilder()
.setId(celStructExprEntry.id())
.setFieldKey(celStructExprEntry.fieldKey())
.setValue(fromCelExpr(celStructExprEntry.value()))
.setOptionalEntry(celStructExprEntry.optionalEntry())
.build());
}
return CreateStruct.newBuilder()
.setMessageName(celCreateStruct.messageName())
.addAllEntries(entries.build())
.build();
}
private static CreateStruct celMapToExprStruct(CelExpr.CelCreateMap celCreateMap) {
ImmutableList.Builder entries = ImmutableList.builder();
for (CelExpr.CelCreateMap.Entry celMapEntry : celCreateMap.entries()) {
CreateStruct.Entry exprMapEntry =
CreateStruct.Entry.newBuilder()
.setId(celMapEntry.id())
.setMapKey(fromCelExpr(celMapEntry.key()))
.setValue(fromCelExpr(celMapEntry.value()))
.setOptionalEntry(celMapEntry.optionalEntry())
.build();
entries.add(exprMapEntry);
}
return CreateStruct.newBuilder().addAllEntries(entries.build()).build();
}
private static ImmutableList fromCelExprList(Iterable celExprList) {
ImmutableList.Builder celExprListBuilder = ImmutableList.builder();
for (CelExpr celExpr : celExprList) {
celExprListBuilder.add(fromCelExpr(celExpr));
}
return celExprListBuilder.build();
}
/**
* Converts a proto-based {@link CelReference} to CEL native representation of {@link Reference}.
*/
public static Reference celReferenceToExprReference(CelReference reference) {
Reference.Builder builder =
Reference.newBuilder().setName(reference.name()).addAllOverloadId(reference.overloadIds());
reference
.value()
.ifPresent(celConstant -> builder.setValue(celConstantToExprConstant(celConstant)));
return builder.build();
}
public static ImmutableMap exprMacroCallsToCelExprMacroCalls(
Map macroCalls) {
return macroCalls.entrySet().stream()
.collect(toImmutableMap(Map.Entry::getKey, v -> fromExpr(v.getValue())));
}
}
// LINT.ThenChange(CelExprV1Alpha1Converter.java)