Skip to content

Commit

Permalink
Address code review comments, added more tests
Browse files Browse the repository at this point in the history
Signed-off-by: Andriy Redko <andriy.redko@aiven.io>
  • Loading branch information
reta committed Mar 28, 2024
1 parent 3632744 commit fdba4b7
Show file tree
Hide file tree
Showing 2 changed files with 270 additions and 0 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,242 @@
/*
* SPDX-License-Identifier: Apache-2.0
*
* The OpenSearch Contributors require contributions made to
* this file be licensed under the Apache-2.0 license or a
* compatible open source license.
*/

package org.opensearch.http.netty4.ssl;

import org.opensearch.common.network.NetworkService;
import org.opensearch.common.settings.ClusterSettings;
import org.opensearch.common.settings.Settings;
import org.opensearch.common.util.MockBigArrays;
import org.opensearch.common.util.MockPageCacheRecycler;
import org.opensearch.core.indices.breaker.NoneCircuitBreakerService;
import org.opensearch.http.HttpServerTransport;
import org.opensearch.http.NullDispatcher;
import org.opensearch.plugins.SecureHttpTransportSettingsProvider;
import org.opensearch.plugins.TransportExceptionHandler;
import org.opensearch.telemetry.tracing.noop.NoopTracer;
import org.opensearch.test.OpenSearchTestCase;
import org.opensearch.threadpool.TestThreadPool;
import org.opensearch.threadpool.ThreadPool;
import org.opensearch.transport.SharedGroupFactory;
import org.opensearch.transport.TransportAdapterProvider;
import org.junit.After;
import org.junit.Before;

import javax.net.ssl.SSLEngine;
import javax.net.ssl.SSLException;

import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Optional;

import io.netty.channel.ChannelInboundHandlerAdapter;

import static org.hamcrest.Matchers.equalTo;

/**
* Tests for the {@link SecureNetty4HttpServerTransport} class.
*/
public class SecureNetty4HttpServerTransportConfigurationTests extends OpenSearchTestCase {

private NetworkService networkService;
private ThreadPool threadPool;
private MockBigArrays bigArrays;
private ClusterSettings clusterSettings;

private static class ConfigurableSecureHttpTransportSettingsProvider implements SecureHttpTransportSettingsProvider {
private final List<TransportAdapterProvider<HttpServerTransport>> transportAdapterProviders;

public ConfigurableSecureHttpTransportSettingsProvider(
List<TransportAdapterProvider<HttpServerTransport>> transportAdapterProviders
) {
this.transportAdapterProviders = transportAdapterProviders;
}

@Override
public Collection<TransportAdapterProvider<HttpServerTransport>> getHttpTransportAdapterProviders(Settings settings) {
return transportAdapterProviders;
}

@Override
public Optional<TransportExceptionHandler> buildHttpServerExceptionHandler(Settings settings, HttpServerTransport transport) {
return Optional.empty();
}

@Override
public Optional<SSLEngine> buildSecureHttpServerEngine(Settings settings, HttpServerTransport transport) throws SSLException {
return Optional.empty();
}
}

@Before
public void setup() throws Exception {
networkService = new NetworkService(Collections.emptyList());
threadPool = new TestThreadPool("test");
bigArrays = new MockBigArrays(new MockPageCacheRecycler(Settings.EMPTY), new NoneCircuitBreakerService());
clusterSettings = new ClusterSettings(Settings.EMPTY, ClusterSettings.BUILT_IN_CLUSTER_SETTINGS);
}

@After
public void shutdown() throws Exception {
if (threadPool != null) {
threadPool.shutdownNow();
}
threadPool = null;
networkService = null;
bigArrays = null;
clusterSettings = null;
}

public void testRequestHeaderVerifier() throws InterruptedException {
final TransportAdapterProvider<HttpServerTransport> transportAdapterProvider = new TransportAdapterProvider<HttpServerTransport>() {
@Override
public String name() {
return SecureNetty4HttpServerTransport.REQUEST_HEADER_VERIFIER;
}

@SuppressWarnings("unchecked")
@Override
public <C> Optional<C> create(Settings settings, HttpServerTransport transport, Class<C> adapterClass) {
return Optional.of((C) new ChannelInboundHandlerAdapter());
}

};

try (
final SecureNetty4HttpServerTransport transport = new SecureNetty4HttpServerTransport(
Settings.EMPTY,
networkService,
bigArrays,
threadPool,
xContentRegistry(),
new NullDispatcher(),
clusterSettings,
new SharedGroupFactory(Settings.EMPTY),
new ConfigurableSecureHttpTransportSettingsProvider(List.of(transportAdapterProvider)),
NoopTracer.INSTANCE
)
) {

}
}

public void testMultipleRequestHeaderVerifiers() throws InterruptedException {
final TransportAdapterProvider<HttpServerTransport> transportAdapterProvider = new TransportAdapterProvider<HttpServerTransport>() {
@Override
public String name() {
return SecureNetty4HttpServerTransport.REQUEST_HEADER_VERIFIER;
}

@SuppressWarnings("unchecked")
@Override
public <C> Optional<C> create(Settings settings, HttpServerTransport transport, Class<C> adapterClass) {
return Optional.of((C) new ChannelInboundHandlerAdapter());
}

};

final IllegalArgumentException ex = assertThrows(
IllegalArgumentException.class,
() -> new SecureNetty4HttpServerTransport(
Settings.EMPTY,
networkService,
bigArrays,
threadPool,
xContentRegistry(),
new NullDispatcher(),
clusterSettings,
new SharedGroupFactory(Settings.EMPTY),
new ConfigurableSecureHttpTransportSettingsProvider(List.of(transportAdapterProvider, transportAdapterProvider)),
NoopTracer.INSTANCE
)
);

assertThat(ex.getMessage(), equalTo("Cannot have more than one header verifier configured, supplied 2"));
}

public void testRequestDecompressor() throws InterruptedException {
final TransportAdapterProvider<HttpServerTransport> transportAdapterProvider = new TransportAdapterProvider<HttpServerTransport>() {
@Override
public String name() {
return SecureNetty4HttpServerTransport.REQUEST_DECOMPRESSOR;
}

@SuppressWarnings("unchecked")
@Override
public <C> Optional<C> create(Settings settings, HttpServerTransport transport, Class<C> adapterClass) {
return Optional.of((C) new ChannelInboundHandlerAdapter());
}

};

try (
final SecureNetty4HttpServerTransport transport = new SecureNetty4HttpServerTransport(
Settings.EMPTY,
networkService,
bigArrays,
threadPool,
xContentRegistry(),
new NullDispatcher(),
clusterSettings,
new SharedGroupFactory(Settings.EMPTY),
new ConfigurableSecureHttpTransportSettingsProvider(List.of(transportAdapterProvider)),
NoopTracer.INSTANCE
)
) {

}
}

public void testRequestDecompressorAndRequestHeaderVerifier() throws InterruptedException {
final TransportAdapterProvider<HttpServerTransport> requestDecompressor = new TransportAdapterProvider<HttpServerTransport>() {
@Override
public String name() {
return SecureNetty4HttpServerTransport.REQUEST_DECOMPRESSOR;
}

@SuppressWarnings("unchecked")
@Override
public <C> Optional<C> create(Settings settings, HttpServerTransport transport, Class<C> adapterClass) {
return Optional.of((C) new ChannelInboundHandlerAdapter());
}

};

final TransportAdapterProvider<HttpServerTransport> requestHeaderVerifier = new TransportAdapterProvider<HttpServerTransport>() {
@Override
public String name() {
return SecureNetty4HttpServerTransport.REQUEST_HEADER_VERIFIER;
}

@SuppressWarnings("unchecked")
@Override
public <C> Optional<C> create(Settings settings, HttpServerTransport transport, Class<C> adapterClass) {
return Optional.of((C) new ChannelInboundHandlerAdapter());
}

};

try (
final SecureNetty4HttpServerTransport transport = new SecureNetty4HttpServerTransport(
Settings.EMPTY,
networkService,
bigArrays,
threadPool,
xContentRegistry(),
new NullDispatcher(),
clusterSettings,
new SharedGroupFactory(Settings.EMPTY),
new ConfigurableSecureHttpTransportSettingsProvider(List.of(requestDecompressor, requestHeaderVerifier)),
NoopTracer.INSTANCE
)
) {

}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,8 @@
import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.Supplier;

import static org.hamcrest.CoreMatchers.startsWith;

public class NetworkModuleTests extends OpenSearchTestCase {
private ThreadPool threadPool;
private SecureSettingsFactory secureSettingsFactory;
Expand Down Expand Up @@ -637,4 +639,30 @@ private NetworkModule newNetworkModule(
secureSettingsFactories
);
}

public void testRegisterSecureTransportMultipleProviers() {
Settings settings = Settings.builder().put(NetworkModule.TRANSPORT_TYPE_KEY, "custom-secure").build();
Supplier<Transport> custom = () -> null; // content doesn't matter we check reference equality
NetworkPlugin plugin = new NetworkPlugin() {
@Override
public Map<String, Supplier<Transport>> getSecureTransports(
Settings settings,
ThreadPool threadPool,
PageCacheRecycler pageCacheRecycler,
CircuitBreakerService circuitBreakerService,
NamedWriteableRegistry namedWriteableRegistry,
NetworkService networkService,
SecureTransportSettingsProvider secureTransportSettingsProvider,
Tracer tracer
) {
return Collections.singletonMap("custom-secure", custom);
}
};

final IllegalArgumentException ex = assertThrows(
IllegalArgumentException.class,
() -> newNetworkModule(settings, null, List.of(secureSettingsFactory, secureSettingsFactory), plugin)
);
assertThat(ex.getMessage(), startsWith("there is more than one secure transport settings provider"));
}
}

0 comments on commit fdba4b7

Please sign in to comment.