All Downloads are FREE. Search and download functionalities are using the official Maven repository.

dev.cel.runtime.RuntimeUnknownResolver Maven / Gradle / Ivy

The newest version!
// 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.runtime;

import com.google.errorprone.annotations.CanIgnoreReturnValue;
import dev.cel.common.annotations.Internal;
import java.util.HashMap;
import java.util.Map;
import java.util.Optional;

/**
 * Runtime representation of the current state in an iterative evaluation session.
 *
 * 

This is used by the interpreter to coordinate identifying and tracking unknown values through * evaluation. * *

CEL Library Internals. Do Not Use. */ @Internal public class RuntimeUnknownResolver { private static final CelAttributeResolver DEFAULT_RESOLVER = new NullAttributeResolver(); /** The underlying resolver for known values. */ private final GlobalResolver resolver; /** Resolver for unknown and resolved attributes. */ private final CelAttributeResolver attributeResolver; private final boolean attributeTrackingEnabled; private RuntimeUnknownResolver( GlobalResolver resolver, CelAttributeResolver attributeResolver, boolean attributeTrackingEnabled) { this.resolver = resolver; this.attributeResolver = attributeResolver; this.attributeTrackingEnabled = attributeTrackingEnabled; } public static RuntimeUnknownResolver fromResolver(GlobalResolver resolver) { // This prevents calculating the attribute trail if it will never be used for // efficiency, but doesn't change observable behavior. return new RuntimeUnknownResolver( resolver, DEFAULT_RESOLVER, /* attributeTrackingEnabled= */ false) {}; } public static Builder builder() { return new Builder(); } /** Builder for {@link RuntimeUnknownResolver}. */ public static class Builder { private CelAttributeResolver attributeResolver; private GlobalResolver resolver; private Builder() { resolver = Activation.EMPTY; attributeResolver = DEFAULT_RESOLVER; } @CanIgnoreReturnValue public Builder setResolver(GlobalResolver resolver) { this.resolver = resolver; return this; } @CanIgnoreReturnValue public Builder setAttributeResolver(CelAttributeResolver resolver) { attributeResolver = resolver; return this; } public RuntimeUnknownResolver build() { return new RuntimeUnknownResolver(resolver, attributeResolver, true); } } /** * Return a single element unknown set if the attribute is partially unknown based on the defined * patterns. */ Optional maybePartialUnknown(CelAttribute attribute) { return attributeResolver.maybePartialUnknown(attribute); } /** Resolve a simple name to a value. */ DefaultInterpreter.IntermediateResult resolveSimpleName(String name, Long exprId) { CelAttribute attr = CelAttribute.EMPTY; if (attributeTrackingEnabled) { attr = CelAttribute.fromQualifiedIdentifier(name); Optional result = attributeResolver.resolve(attr); if (result.isPresent()) { return DefaultInterpreter.IntermediateResult.create(attr, result.get()); } } Object result = resolver.resolve(name); return DefaultInterpreter.IntermediateResult.create( attr, InterpreterUtil.valueOrUnknown(result, exprId)); } void cacheLazilyEvaluatedResult(String name, DefaultInterpreter.IntermediateResult result) { // no-op. Caching is handled in ScopedResolver. } /** * Attempt to resolve an attribute bound to a context variable. This is used to shadow lazily * resolved values behind field accesses and index operations. */ Optional resolveAttribute(CelAttribute attr) { return attributeResolver.resolve(attr); } ScopedResolver withScope(Map vars) { return new ScopedResolver(this, vars); } static final class ScopedResolver extends RuntimeUnknownResolver { private final RuntimeUnknownResolver parent; private final Map shadowedVars; private final Map lazyEvalResultCache; private ScopedResolver( RuntimeUnknownResolver parent, Map shadowedVars) { super(parent.resolver, parent.attributeResolver, parent.attributeTrackingEnabled); this.parent = parent; this.shadowedVars = shadowedVars; this.lazyEvalResultCache = new HashMap<>(); } @Override DefaultInterpreter.IntermediateResult resolveSimpleName(String name, Long exprId) { DefaultInterpreter.IntermediateResult result = lazyEvalResultCache.get(name); if (result != null) { return result; } result = shadowedVars.get(name); if (result != null) { return result; } return parent.resolveSimpleName(name, exprId); } @Override void cacheLazilyEvaluatedResult(String name, DefaultInterpreter.IntermediateResult result) { lazyEvalResultCache.put(name, result); } } /** Null implementation for attribute resolution. */ private static final class NullAttributeResolver implements CelAttributeResolver { @Override public Optional resolve(CelAttribute attr) { return Optional.empty(); } @Override public Optional maybePartialUnknown(CelAttribute attr) { return Optional.empty(); } } }