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

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

// 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 static java.lang.Math.min;

import com.google.auto.value.AutoValue;
import com.google.common.base.Preconditions;
import com.google.common.base.Splitter;
import com.google.common.collect.ImmutableList;
import com.google.errorprone.annotations.Immutable;

/**
 * A Pattern for matching against {@link CelAttribute}s.
 *
 * 

CelAttributePatterns are structured the same as attributes, but permit some qualifiers be * replaced with wildcards. */ @Immutable @AutoValue public abstract class CelAttributePattern { /** Constructs a CelAttributePattern from a list of qualifiers. */ public static CelAttributePattern create(ImmutableList qualifiers) { Preconditions.checkArgument(!qualifiers.isEmpty(), "qualifiers must be non-empty"); Preconditions.checkArgument( CelAttribute.Qualifier.isLegalIdentifier(qualifiers.get(0)), "'%s' is not a legal CEL identifier", qualifiers.get(0)); return new AutoValue_CelAttributePattern(qualifiers); } /** Constructs a pattern from a single root identifier (single string qualifier). */ public static CelAttributePattern create(String rootIdentifier) { Preconditions.checkArgument( CelAttribute.Qualifier.isLegalIdentifier(rootIdentifier), "'%s' is not a legal CEL identifier", rootIdentifier); return new AutoValue_CelAttributePattern( ImmutableList.of(CelAttribute.Qualifier.ofString(rootIdentifier))); } /** * Attempts to parse a dot qualified identifier into a CEL attribute. * * @throws IllegalArgumentException if qualifiedIdentifier isn't a legal qualified identifier. * Note: this is intended for use with reference names in a checked CEL expression -- it does * not check for some edge cases (e.g. reserved words). */ public static CelAttributePattern fromQualifiedIdentifier(String qualifiedIdentifier) { ImmutableList.Builder qualifiers = ImmutableList.builder(); Splitter.on(".") .split(qualifiedIdentifier) .forEach((String element) -> qualifiers.add(CelAttribute.Qualifier.ofString(element))); return new AutoValue_CelAttributePattern(qualifiers.build()); } /** The list of qualifiers representing the select paths this pattern matches. */ public abstract ImmutableList qualifiers(); /** Create a new attribute pattern that specifies a subfield of this pattern. */ public CelAttributePattern qualify(CelAttribute.Qualifier qualifier) { return new AutoValue_CelAttributePattern( ImmutableList.builder() .addAll(qualifiers()) .add(qualifier) .build()); } @Override public final String toString() { Preconditions.checkState( !qualifiers().isEmpty() && qualifiers().get(0).kind() == CelAttribute.Qualifier.Kind.AS_STRING, "CelAttribute must have a root qualifier that is a legal identifier"); StringBuilder cname = new StringBuilder(qualifiers().get(0).asString()); for (CelAttribute.Qualifier qualifier : qualifiers().subList(1, qualifiers().size())) { switch (qualifier.kind()) { case WILD_CARD: cname.append(".*"); break; case AS_STRING: if (CelAttribute.Qualifier.isLegalIdentifier(qualifier)) { cname.append(".").append(qualifier.asString()); } else { cname.append(qualifier.toIndexFormat()); } break; default: cname.append(qualifier.toIndexFormat()); break; } } return cname.toString(); } /** * Return whether this pattern matches the given attribute. * *

A pattern matches an attribute if the pattern contains the attribute (e.g. fully matches the * attribute or one of its parents). */ public boolean isMatch(CelAttribute attribute) { if (attribute.equals(CelAttribute.EMPTY)) { return false; } int qualifierCount = min(qualifiers().size(), attribute.qualifiers().size()); int i = 0; for (; i < qualifierCount; i++) { CelAttribute.Qualifier pattern = qualifiers().get(i); CelAttribute.Qualifier qualifier = attribute.qualifiers().get(i); if (pattern.kind() != CelAttribute.Qualifier.Kind.WILD_CARD && !pattern.equals(qualifier)) { return false; } } // If the end of the loop is reached, either the pattern was exhausted (full match) or // the attribute was exhausted (partial match / a sub field was matched). return i >= qualifiers().size(); } /** * Return whether this pattern matches the given attribute or any of its descendants. * *

A partial match indicates that the attribute may contain some missing data, so the complete * object is may be in an undefined state. For example: * *

    *
  • pattern (this) object.field.list_field *
  • attribute (arg) object.field *
* * is not a full match, but is a partial match. */ public boolean isPartialMatch(CelAttribute attribute) { if (attribute.equals(CelAttribute.EMPTY)) { return false; } int qualifierCount = min(qualifiers().size(), attribute.qualifiers().size()); for (int i = 0; i < qualifierCount; i++) { CelAttribute.Qualifier pattern = qualifiers().get(i); CelAttribute.Qualifier qualifier = attribute.qualifiers().get(i); if (pattern.kind() != CelAttribute.Qualifier.Kind.WILD_CARD && !pattern.equals(qualifier)) { return false; } } // If the end of the loop is reached, either the pattern was exhausted (full match) or // the attribute was exhausted (partial match / a sub field was matched). return true; } /** * Return an appropriate attribute for a pattern match. * *

For a partial match, return the attribute. * *

For a full match, return the (possibly parent attribute) */ public CelAttribute simplify(CelAttribute candidate) { int patternLen = this.qualifiers().size(); if (patternLen < candidate.qualifiers().size()) { return CelAttribute.create(candidate.qualifiers().subList(0, patternLen)); } return candidate; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy