PrefixingDelegatingAggregationOperationContext.java
/*
* Copyright 2018-present the original author or authors.
*
* 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 org.springframework.data.mongodb.core.aggregation;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.bson.Document;
import org.bson.codecs.configuration.CodecRegistry;
import org.jspecify.annotations.Nullable;
import org.springframework.data.mongodb.core.aggregation.ExposedFields.FieldReference;
/**
* {@link AggregationOperationContext} implementation prefixing non-command keys on root level with the given prefix.
* Useful when mapping fields to domain specific types while having to prefix keys for query purpose.
* <br />
* Fields to be excluded from prefixing my be added to a {@literal denylist}.
*
* @author Christoph Strobl
* @author Mark Paluch
* @since 2.1
*/
public class PrefixingDelegatingAggregationOperationContext implements AggregationOperationContext {
private final AggregationOperationContext delegate;
private final String prefix;
private final Set<String> denylist;
public PrefixingDelegatingAggregationOperationContext(AggregationOperationContext delegate, String prefix) {
this(delegate, prefix, Collections.emptySet());
}
public PrefixingDelegatingAggregationOperationContext(AggregationOperationContext delegate, String prefix,
Collection<String> denylist) {
this.delegate = delegate;
this.prefix = prefix;
this.denylist = new HashSet<>(denylist);
}
@Override
public Document getMappedObject(Document document) {
return doPrefix(delegate.getMappedObject(document));
}
@Override
public Document getMappedObject(Document document, @Nullable Class<?> type) {
return doPrefix(delegate.getMappedObject(document, type));
}
@Override
public FieldReference getReference(Field field) {
return delegate.getReference(field);
}
@Override
public FieldReference getReference(String name) {
return delegate.getReference(name);
}
@Override
public Fields getFields(Class<?> type) {
return delegate.getFields(type);
}
@Override
public CodecRegistry getCodecRegistry() {
return delegate.getCodecRegistry();
}
@SuppressWarnings("unchecked")
private Document doPrefix(Document source) {
Document result = new Document();
for (Map.Entry<String, Object> entry : source.entrySet()) {
String key = prefixKey(entry.getKey());
Object value = entry.getValue();
if (entry.getValue() instanceof Collection) {
Collection<Object> sourceCollection = (Collection<Object>) entry.getValue();
value = prefixCollection(sourceCollection);
}
result.append(key, value);
}
return result;
}
private String prefixKey(String key) {
return (key.startsWith("$") || isDenied(key)) ? key : (prefix + "." + key);
}
private Object prefixCollection(Collection<Object> sourceCollection) {
List<Object> prefixed = new ArrayList<>(sourceCollection.size());
for (Object o : sourceCollection) {
if (o instanceof Document document) {
prefixed.add(doPrefix(document));
} else {
prefixed.add(o);
}
}
return prefixed;
}
private boolean isDenied(String key) {
if (denylist.contains(key)) {
return true;
}
if (!key.contains(".")) {
return false;
}
for (String denied : denylist) {
if (key.startsWith(denied + ".")) {
return true;
}
}
return false;
}
}