io.helidon.tracing.MapHeaderConsumer Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of helidon-tracing Show documentation
Show all versions of helidon-tracing Show documentation
Support for distributed tracing.
/*
* Copyright (c) 2022 Oracle and/or its affiliates.
*
* 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
*
* http://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 io.helidon.tracing;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.TreeMap;
import java.util.concurrent.ConcurrentSkipListMap;
import java.util.stream.Stream;
/**
* Side-effect-safe implementation of {@link HeaderConsumer}.
*
* This class has historically modified the map that is passed as the initial headers.
* If that map was not case-insensitive (as headers should be), then this class would not honor
* case-insensitive conventions.
*
*
* To implement case-insensitivity when the provided headers map is not itself case-insensitive,
* this class maintains a "shadow" map and uses that for key-based operations that must not depend
* on case.
*
*
* For efficiency, callers should provide a case-insensitive map as the initial headers.
*
*/
class MapHeaderConsumer implements HeaderConsumer {
private final Map> delegate;
private final Map> caseInsensitiveShadow;
private final Map> mapToQuery;
MapHeaderConsumer(Map> headers) {
delegate = headers;
if (isCaseInsensitive(headers)) {
caseInsensitiveShadow = null;
mapToQuery = delegate;
} else {
caseInsensitiveShadow = new ConcurrentSkipListMap<>(String.CASE_INSENSITIVE_ORDER);
caseInsensitiveShadow.putAll(headers);
mapToQuery = caseInsensitiveShadow;
}
}
@Override
public Iterable keys() {
return delegate.keySet();
}
@Override
public Optional get(String key) {
return Optional.ofNullable(mapToQuery.get(key))
.map(List::stream)
.flatMap(Stream::findFirst);
}
@Override
public Iterable getAll(String key) {
return mapToQuery.getOrDefault(key, List.of());
}
@Override
public boolean contains(String key) {
return mapToQuery.containsKey(key);
}
@Override
public void setIfAbsent(String key, String... values) {
List valueList = List.of(values);
delegate.putIfAbsent(key, valueList);
if (caseInsensitiveShadow != null) {
caseInsensitiveShadow.putIfAbsent(key, valueList);
}
}
@Override
public void set(String key, String... values) {
List valueList = List.of(values);
delegate.put(key, valueList);
if (caseInsensitiveShadow != null) {
caseInsensitiveShadow.put(key, valueList);
}
}
private static boolean isCaseInsensitive(Map> headers) {
return (headers instanceof TreeMap
&& ((TreeMap, ?>) headers).comparator() == String.CASE_INSENSITIVE_ORDER)
|| (headers instanceof ConcurrentSkipListMap, ?>
&& ((ConcurrentSkipListMap, ?>) headers).comparator() == String.CASE_INSENSITIVE_ORDER);
}
}