Ensure ChannelOptions are applied in the same order as configured in Http2StreamChannelBootstrap (#9998) (#10001)

Motivation:

https://github.com/netty/netty/pull/9848 changed how we handled ChannelOptions internally to use a ConcurrentHashMap. This unfortunally had the side-effect that the ordering may be affected and not stable anymore. Here the problem is that sometimes we do validation based on two different ChannelOptions (for example we validate high and low watermarks against each other). Thus even if the user specified the options in the same order we may fail to configure them.

Modifications:

- Use again a LinkedHashMap to preserve order

Result:

Apply ChannelOptions in correct and expected order
This commit is contained in:
Norman Maurer 2020-02-07 09:14:16 +01:00
parent e7e999373f
commit 98cf4cf565

View File

@ -34,6 +34,7 @@ import io.netty.util.internal.logging.InternalLogger;
import io.netty.util.internal.logging.InternalLoggerFactory; import io.netty.util.internal.logging.InternalLoggerFactory;
import java.nio.channels.ClosedChannelException; import java.nio.channels.ClosedChannelException;
import java.util.LinkedHashMap;
import java.util.Map; import java.util.Map;
import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentHashMap;
@ -45,7 +46,9 @@ public final class Http2StreamChannelBootstrap {
@SuppressWarnings("unchecked") @SuppressWarnings("unchecked")
private static final Map.Entry<AttributeKey<?>, Object>[] EMPTY_ATTRIBUTE_ARRAY = new Map.Entry[0]; private static final Map.Entry<AttributeKey<?>, Object>[] EMPTY_ATTRIBUTE_ARRAY = new Map.Entry[0];
private final Map<ChannelOption<?>, Object> options = new ConcurrentHashMap<>(); // The order in which ChannelOptions are applied is important they may depend on each other for validation
// purposes.
private final Map<ChannelOption<?>, Object> options = new LinkedHashMap<>();
private final Map<AttributeKey<?>, Object> attrs = new ConcurrentHashMap<>(); private final Map<AttributeKey<?>, Object> attrs = new ConcurrentHashMap<>();
private final Channel channel; private final Channel channel;
private volatile ChannelHandler handler; private volatile ChannelHandler handler;
@ -64,11 +67,14 @@ public final class Http2StreamChannelBootstrap {
@SuppressWarnings("unchecked") @SuppressWarnings("unchecked")
public <T> Http2StreamChannelBootstrap option(ChannelOption<T> option, T value) { public <T> Http2StreamChannelBootstrap option(ChannelOption<T> option, T value) {
requireNonNull(option, "option"); requireNonNull(option, "option");
synchronized (options) {
if (value == null) { if (value == null) {
options.remove(option); options.remove(option);
} else { } else {
options.put(option, value); options.put(option, value);
} }
}
return this; return this;
} }
@ -195,7 +201,12 @@ public final class Http2StreamChannelBootstrap {
if (handler != null) { if (handler != null) {
p.addLast(handler); p.addLast(handler);
} }
setChannelOptions(channel, options.entrySet().toArray(EMPTY_OPTION_ARRAY)); final Map.Entry<ChannelOption<?>, Object> [] optionArray;
synchronized (options) {
optionArray = options.entrySet().toArray(EMPTY_OPTION_ARRAY);
}
setChannelOptions(channel, optionArray);
setAttributes(channel, attrs.entrySet().toArray(EMPTY_ATTRIBUTE_ARRAY)); setAttributes(channel, attrs.entrySet().toArray(EMPTY_ATTRIBUTE_ARRAY));
} }