dev.cel.common.CelDescriptorUtil 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 com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.protobuf.DescriptorProtos.FileDescriptorSet;
import com.google.protobuf.Descriptors.Descriptor;
import com.google.protobuf.Descriptors.FieldDescriptor;
import com.google.protobuf.Descriptors.FileDescriptor;
import com.google.protobuf.Descriptors.GenericDescriptor;
import dev.cel.common.internal.FileDescriptorSetConverter;
import dev.cel.common.types.CelTypes;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashSet;
import java.util.Set;
/** Utility class for working with protobuf descriptors. */
public final class CelDescriptorUtil {
private CelDescriptorUtil() {}
/**
* Converts descriptor collection to an ImmutableMap.
*
* Key: Descriptor's full name, Value: Descriptor object
*/
public static ImmutableMap descriptorCollectionToMap(
Collection descriptors) {
ImmutableMap.Builder descriptorMapBuilder = new ImmutableMap.Builder<>();
descriptors.forEach(d -> descriptorMapBuilder.put(d.getFullName(), d));
return descriptorMapBuilder.buildOrThrow();
}
/**
* Get the full {@code FileDescriptor} set needed to accurately instantiate the {@code
* descriptors}.
*/
public static ImmutableSet getFileDescriptorsForDescriptors(
Iterable descriptors) {
ImmutableSet.Builder fileDescriptors = ImmutableSet.builder();
descriptors.forEach(d -> fileDescriptors.add(d.getFile()));
return getFileDescriptorsAndDependencies(fileDescriptors.build());
}
/**
* Convert a {@code FileDescriptorSet} into a set of {@code FileDescriptor} instances.
*
* Warning: This will produce unique FileDescriptor instances. Use with care especially in
* hermetic environments.
*/
public static ImmutableSet getFileDescriptorsFromFileDescriptorSet(
FileDescriptorSet fileDescriptorSet) {
return FileDescriptorSetConverter.convert(fileDescriptorSet);
}
/**
* Extract the full message {@code CelDescriptors} set from the input set of {@code
* fileDescriptors}. All message type, enum, extension and file descriptors will be extracted.
*
* Note, the input {@code fileDescriptors} set will be expanded to the complete set of
* dependencies needed to describe the types within the provided files.
*/
public static CelDescriptors getAllDescriptorsFromFileDescriptor(
FileDescriptor... fileDescriptors) {
return getAllDescriptorsFromFileDescriptor(Arrays.asList(fileDescriptors));
}
/**
* Extract the full message {@code CelDescriptors} set from the input set of {@code
* fileDescriptors}. All message type, enum, extension and file descriptors will be extracted.
*
*
Note, the input {@code fileDescriptors} set will be expanded to the complete set of
* dependencies needed to describe the types within the provided files.
*/
public static CelDescriptors getAllDescriptorsFromFileDescriptor(
Iterable fileDescriptors) {
return getAllDescriptorsFromFileDescriptor(fileDescriptors, true);
}
/**
* Extract the full message {@code FileDescriptor} set from the input set of {@code
* fileDescriptors}. All message type, enum, extension and file descriptors will be extracted.
*
* @param resolveTypeDependencies Performs a deep type dependency resolution by expanding all the
* FileDescriptors marked as dependents listed in their imports (Ex: If FileDescriptor A
* imports on FileDescriptor B, FD B's descriptors will be pulled in). Setting false will
* disable this.
*/
public static CelDescriptors getAllDescriptorsFromFileDescriptor(
Iterable fileDescriptors, boolean resolveTypeDependencies) {
ImmutableSet allFileDescriptors =
resolveTypeDependencies
? getFileDescriptorsAndDependencies(fileDescriptors)
: ImmutableSet.copyOf(fileDescriptors);
CelDescriptors.Builder celDescriptorsBuilder = CelDescriptors.builder();
allFileDescriptors.forEach(
fd -> collectAllDescriptorsFromFileDescriptor(fd, celDescriptorsBuilder));
return celDescriptorsBuilder.build();
}
/**
* Collects all descriptors referenced by the {@code FileDescriptor}, including message types,
* enums and extensions.
*/
private static void collectAllDescriptorsFromFileDescriptor(
FileDescriptor fileDescriptor, CelDescriptors.Builder celDescriptors) {
celDescriptors.addFileDescriptors(fileDescriptor);
collectEnumDescriptors(fileDescriptor, celDescriptors);
collectMessageTypeDescriptors(fileDescriptor, celDescriptors);
collectExtensions(fileDescriptor, celDescriptors);
}
private static void collectEnumDescriptors(
FileDescriptor fileDescriptor, CelDescriptors.Builder celDescriptors) {
Set visitedDescriptors = new HashSet<>();
celDescriptors.addEnumDescriptors(fileDescriptor.getEnumTypes());
fileDescriptor
.getMessageTypes()
.forEach(d -> collectEnumDescriptors(d, visitedDescriptors, celDescriptors));
fileDescriptor.getExtensions().forEach(ext -> collectEnumDescriptors(ext, celDescriptors));
}
private static void collectEnumDescriptors(
Descriptor descriptor,
Set visitedDescriptors,
CelDescriptors.Builder celDescriptors) {
if (visitedDescriptors.contains(descriptor.getFullName())) {
return;
}
visitedDescriptors.add(descriptor.getFullName());
celDescriptors.addEnumDescriptors(descriptor.getEnumTypes());
descriptor.getFields().forEach(field -> collectEnumDescriptors(field, celDescriptors));
descriptor
.getNestedTypes()
.forEach(nested -> collectEnumDescriptors(nested, visitedDescriptors, celDescriptors));
}
private static void collectEnumDescriptors(
FieldDescriptor fieldDescriptor, CelDescriptors.Builder celDescriptors) {
if (!fieldDescriptor.getType().equals(FieldDescriptor.Type.ENUM)) {
return;
}
celDescriptors.addEnumDescriptors(fieldDescriptor.getEnumType());
}
private static void collectMessageTypeDescriptors(
FileDescriptor descriptor, CelDescriptors.Builder celDescriptors) {
Set visited = new HashSet<>();
descriptor
.getMessageTypes()
.forEach((d) -> collectMessageTypeDescriptors(d, visited, celDescriptors));
}
private static void collectMessageTypeDescriptors(
Descriptor descriptor, Set visited, CelDescriptors.Builder celDescriptors) {
String messageName = descriptor.getFullName();
if (visited.contains(messageName)) {
return;
}
if (!descriptor.getOptions().getMapEntry()) {
visited.add(messageName);
celDescriptors.addMessageTypeDescriptors(descriptor);
}
if (CelTypes.getWellKnownCelType(messageName).isPresent()) {
return;
}
descriptor
.getNestedTypes()
.forEach(nested -> collectMessageTypeDescriptors(nested, visited, celDescriptors));
descriptor
.getFields()
.forEach(field -> collectFieldDescriptors(field, visited, celDescriptors));
celDescriptors.addExtensionDescriptors(descriptor.getExtensions());
}
private static void collectExtensions(
FileDescriptor fileDescriptor, CelDescriptors.Builder celDescriptors) {
celDescriptors.addExtensionDescriptors(fileDescriptor.getExtensions());
Set visited = new HashSet<>();
fileDescriptor
.getExtensions()
.forEach((d) -> collectFieldDescriptors(d, visited, celDescriptors));
}
/**
* Enumerates all message types in a field descriptor. This is used for handling nested type
* entries within a message.
*/
private static void collectFieldDescriptors(
FieldDescriptor fieldDescriptor, Set visited, CelDescriptors.Builder celDescriptors) {
if (!fieldDescriptor.getType().equals(FieldDescriptor.Type.MESSAGE)) {
return;
}
Descriptor fieldMessageType = fieldDescriptor.getMessageType();
if (fieldDescriptor.isMapField()) {
FieldDescriptor valueDescriptor = fieldDescriptor.getMessageType().getFields().get(1);
if (valueDescriptor.getType().equals(FieldDescriptor.Type.MESSAGE)) {
collectMessageTypeDescriptors(valueDescriptor.getMessageType(), visited, celDescriptors);
}
} else {
collectMessageTypeDescriptors(fieldMessageType, visited, celDescriptors);
}
}
/**
* Get the full {@code FileDescriptor} set needed by the referenced {@code fileDescriptors}.
*
* If the {@code FileDescriptor} is provided which contains the {@code AttributeContext}
* message definition, then the expanded {@code FileDescriptor} set should also include the files
* defining {@code Duration}, {@code Timestamp}, and {@code Struct}.
*/
private static ImmutableSet getFileDescriptorsAndDependencies(
Iterable fileDescriptors) {
Set visited = new HashSet<>();
ImmutableSet.Builder expandedFileDescriptors = ImmutableSet.builder();
ImmutableSet.copyOf(fileDescriptors)
.forEach((fd) -> copyToFileDescriptorSet(visited, fd, expandedFileDescriptors));
return expandedFileDescriptors.build();
}
private static void copyToFileDescriptorSet(
Set visited, FileDescriptor fd, ImmutableSet.Builder files) {
if (visited.contains(fd.getFullName())) {
return;
}
visited.add(fd.getFullName());
for (FileDescriptor dep : fd.getDependencies()) {
copyToFileDescriptorSet(visited, dep, files);
}
files.add(fd);
}
}