reactor.util.context.Context Maven / Gradle / Ivy
Show all versions of redisson-all Show documentation
/*
* Copyright (c) 2017-2021 VMware Inc. or its affiliates, All Rights Reserved.
*
* 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 reactor.util.context;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Objects;
import java.util.Optional;
import reactor.util.annotation.Nullable;
/**
* A key/value store that is propagated between components such as operators via the
* context protocol. Contexts are ideal to transport orthogonal information such as
* tracing or security tokens.
*
* {@link Context} implementations are thread-safe and immutable: mutative operations like
* {@link #put(Object, Object)} will in fact return a new {@link Context} instance.
*
* Note that contexts are optimized for low cardinality key/value storage, and a user
* might want to associate a dedicated mutable structure to a single key to represent his
* own context instead of using multiple {@link #put}, which could be more costly.
* Past five user key/value pair, the {@link Context} will use a copy-on-write
* implementation backed by a new {@link java.util.Map} on each {@link #put}.
*
* @author Stephane Maldini
*/
public interface Context extends ContextView {
/**
* Return an empty {@link Context}
*
* @return an empty {@link Context}
*/
static Context empty() {
return Context0.INSTANCE;
}
/**
* Create a {@link Context} pre-initialized with one key-value pair.
*
* @param key the key to initialize.
* @param value the value for the key.
* @return a {@link Context} with a single entry.
* @throws NullPointerException if either key or value are null
*/
static Context of(Object key, Object value) {
return new Context1(key, value);
}
/**
* Create a {@link Context} pre-initialized with two key-value pairs.
*
* @param key1 the first key to initialize.
* @param value1 the value for the first key.
* @param key2 the second key to initialize.
* @param value2 the value for the second key.
* @return a {@link Context} with two entries.
* @throws NullPointerException if any key or value is null
*/
static Context of(Object key1, Object value1,
Object key2, Object value2) {
return new Context2(key1, value1, key2, value2);
}
/**
* Create a {@link Context} pre-initialized with three key-value pairs.
*
* @param key1 the first key to initialize.
* @param value1 the value for the first key.
* @param key2 the second key to initialize.
* @param value2 the value for the second key.
* @param key3 the third key to initialize.
* @param value3 the value for the third key.
* @return a {@link Context} with three entries.
* @throws NullPointerException if any key or value is null
*/
static Context of(Object key1, Object value1,
Object key2, Object value2,
Object key3, Object value3) {
return new Context3(key1, value1, key2, value2, key3, value3);
}
/**
* Create a {@link Context} pre-initialized with four key-value pairs.
*
* @param key1 the first key to initialize.
* @param value1 the value for the first key.
* @param key2 the second key to initialize.
* @param value2 the value for the second key.
* @param key3 the third key to initialize.
* @param value3 the value for the third key.
* @param key4 the fourth key to initialize.
* @param value4 the value for the fourth key.
* @return a {@link Context} with four entries.
* @throws NullPointerException if any key or value is null
*/
static Context of(Object key1, Object value1,
Object key2, Object value2,
Object key3, Object value3,
Object key4, Object value4) {
return new Context4(key1, value1, key2, value2, key3, value3, key4, value4);
}
/**
* Create a {@link Context} pre-initialized with five key-value pairs.
*
* @param key1 the first key to initialize.
* @param value1 the value for the first key.
* @param key2 the second key to initialize.
* @param value2 the value for the second key.
* @param key3 the third key to initialize.
* @param value3 the value for the third key.
* @param key4 the fourth key to initialize.
* @param value4 the value for the fourth key.
* @param key5 the fifth key to initialize.
* @param value5 the value for the fifth key.
* @return a {@link Context} with five entries.
* @throws NullPointerException if any key or value is null
*/
static Context of(Object key1, Object value1,
Object key2, Object value2,
Object key3, Object value3,
Object key4, Object value4,
Object key5, Object value5) {
return new Context5(key1, value1, key2, value2, key3, value3, key4, value4, key5, value5);
}
/**
* Create a {@link Context} out of a {@link Map}. Prefer this method if you're somehow
* incapable of checking keys are all distinct in other {@link #of(Object, Object, Object, Object, Object, Object, Object, Object)}
* implementations.
*
* @implNote this method compacts smaller maps into a relevant fields-based implementation
* when map size is less than 6.
*/
static Context of(Map, ?> map) {
int size = Objects.requireNonNull(map, "map").size();
if (size == 0) return Context.empty();
if (size <= 5) {
Map.Entry[] entries = map.entrySet().toArray(new Map.Entry[size]);
switch (size) {
case 1:
return new Context1(entries[0].getKey(), entries[0].getValue());
case 2:
return new Context2(entries[0].getKey(), entries[0].getValue(),
entries[1].getKey(), entries[1].getValue());
case 3:
return new Context3(entries[0].getKey(), entries[0].getValue(),
entries[1].getKey(), entries[1].getValue(),
entries[2].getKey(), entries[2].getValue());
case 4:
return new Context4(entries[0].getKey(), entries[0].getValue(),
entries[1].getKey(), entries[1].getValue(),
entries[2].getKey(), entries[2].getValue(),
entries[3].getKey(), entries[3].getValue());
case 5:
return new Context5(entries[0].getKey(), entries[0].getValue(),
entries[1].getKey(), entries[1].getValue(),
entries[2].getKey(), entries[2].getValue(),
entries[3].getKey(), entries[3].getValue(),
entries[4].getKey(), entries[4].getValue());
}
}
// Since ContextN(Map) is a low level API that DOES NOT perform null checks,
// we need to check every key/value before passing it to ContextN(Map)
map.forEach((key, value) -> {
Objects.requireNonNull(key, "null key found");
if (value == null) {
throw new NullPointerException("null value for key " + key);
}
});
@SuppressWarnings("unchecked")
final Map