
com.hazelcast.internal.tpcengine.nio.NioAsyncSocketBuilder Maven / Gradle / Ivy
The newest version!
/*
* Copyright (c) 2008-2024, Hazelcast, Inc. 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
*
* 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 com.hazelcast.internal.tpcengine.nio;
import com.hazelcast.internal.tpcengine.Option;
import com.hazelcast.internal.tpcengine.net.AsyncSocket;
import com.hazelcast.internal.tpcengine.net.AsyncSocketBuilder;
import com.hazelcast.internal.tpcengine.net.AsyncSocketReader;
import com.hazelcast.internal.tpcengine.net.AsyncSocketWriter;
import java.io.IOException;
import java.io.UncheckedIOException;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.nio.channels.SocketChannel;
import java.util.concurrent.CompletableFuture;
import static com.hazelcast.internal.tpcengine.net.AsyncSocketOptions.SSL_ENGINE_FACTORY;
import static com.hazelcast.internal.tpcengine.util.ExceptionUtil.sneakyThrow;
import static com.hazelcast.internal.tpcengine.util.Preconditions.checkNotNull;
import static com.hazelcast.internal.tpcengine.util.Preconditions.checkPositive;
/**
* A {@link AsyncSocketBuilder} specific to the {@link NioAsyncSocket}.
*/
public class NioAsyncSocketBuilder implements AsyncSocketBuilder {
static final int DEFAULT_WRITE_QUEUE_CAPACITY = 2 << 16;
private static final Constructor TLS_NIO_ASYNC_SOCKET_CONSTRUCTOR;
private static final String TLS_NIO_ASYNC_SOCKET_CLASS_NAME = "com.hazelcast.internal.tpcengine.nio.TlsNioAsyncSocket";
static {
Constructor tlsNioAsyncSocketConstructor = null;
try {
Class> clazz = NioAsyncSocketBuilder.class.getClassLoader().loadClass(TLS_NIO_ASYNC_SOCKET_CLASS_NAME);
tlsNioAsyncSocketConstructor = (Constructor) clazz.getDeclaredConstructor(NioAsyncSocketBuilder.class);
tlsNioAsyncSocketConstructor.setAccessible(true);
} catch (ClassNotFoundException e) {
tlsNioAsyncSocketConstructor = null;
} catch (NoSuchMethodException e) {
throw new Error(e);
} finally {
TLS_NIO_ASYNC_SOCKET_CONSTRUCTOR = tlsNioAsyncSocketConstructor;
}
}
final NioReactor reactor;
final SocketChannel socketChannel;
final NioAcceptRequest acceptRequest;
final boolean clientSide;
boolean directBuffers = true;
int writeQueueCapacity = DEFAULT_WRITE_QUEUE_CAPACITY;
AsyncSocketReader reader;
AsyncSocketWriter writer;
NioAsyncSocketOptions options;
private boolean built;
NioAsyncSocketBuilder(NioReactor reactor, NioAcceptRequest acceptRequest) {
try {
this.reactor = reactor;
this.acceptRequest = acceptRequest;
if (acceptRequest == null) {
this.socketChannel = SocketChannel.open();
this.clientSide = true;
} else {
this.socketChannel = acceptRequest.socketChannel;
this.clientSide = false;
}
this.socketChannel.configureBlocking(false);
this.options = new NioAsyncSocketOptions(socketChannel);
} catch (IOException e) {
throw new UncheckedIOException(e);
}
}
@Override
public NioAsyncSocketBuilder set(Option option, T value) {
verifyNotBuilt();
options.set(option, value);
return this;
}
@Override
public boolean setIfSupported(Option option, T value) {
verifyNotBuilt();
return options.set(option, value);
}
public NioAsyncSocketBuilder setDirectBuffers(boolean directBuffers) {
verifyNotBuilt();
this.directBuffers = directBuffers;
return this;
}
public NioAsyncSocketBuilder setWriteQueueCapacity(int writeQueueCapacity) {
verifyNotBuilt();
this.writeQueueCapacity = checkPositive(writeQueueCapacity, "writeQueueCapacity");
return this;
}
@Override
public final NioAsyncSocketBuilder setReader(AsyncSocketReader reader) {
verifyNotBuilt();
this.reader = checkNotNull(reader);
return this;
}
@Override
public AsyncSocketBuilder setWriter(AsyncSocketWriter writer) {
verifyNotBuilt();
this.writer = checkNotNull(writer);
return this;
}
@SuppressWarnings("java:S1181")
@Override
public AsyncSocket build() {
verifyNotBuilt();
built = true;
if (reader == null) {
throw new IllegalStateException("reader is not configured.");
}
if (Thread.currentThread() == reactor.eventloopThread()) {
return options.get(SSL_ENGINE_FACTORY) == null
? new NioAsyncSocket(this)
: newTlsNioAsyncSocket(this);
} else {
CompletableFuture future = new CompletableFuture<>();
reactor.execute(() -> {
try {
AsyncSocket asyncSocket = options.get(SSL_ENGINE_FACTORY) == null
? new NioAsyncSocket(NioAsyncSocketBuilder.this)
: newTlsNioAsyncSocket(NioAsyncSocketBuilder.this);
future.complete(asyncSocket);
} catch (Throwable e) {
future.completeExceptionally(e);
throw sneakyThrow(e);
}
});
return future.join();
}
}
private AsyncSocket newTlsNioAsyncSocket(NioAsyncSocketBuilder nioAsyncSocketBuilder) {
if (TLS_NIO_ASYNC_SOCKET_CONSTRUCTOR == null) {
throw new IllegalStateException("class " + TLS_NIO_ASYNC_SOCKET_CLASS_NAME + " is not found");
}
try {
return TLS_NIO_ASYNC_SOCKET_CONSTRUCTOR.newInstance(nioAsyncSocketBuilder);
} catch (InvocationTargetException | InstantiationException | IllegalAccessException e) {
throw new RuntimeException(e);
}
}
private void verifyNotBuilt() {
if (built) {
throw new IllegalStateException("Can't call build twice on the same AsyncSocketBuilder");
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy