org.bitcoinj.net.NioClient Maven / Gradle / Ivy
/*
* Copyright 2013 Google Inc.
* Copyright 2014 Andreas Schildbach
*
* 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 org.bitcoinj.net;
import com.google.common.base.Throwables;
import org.bitcoinj.utils.ListenableCompletableFuture;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.IOException;
import java.net.SocketAddress;
import java.nio.ByteBuffer;
import java.time.Duration;
/**
* Creates a simple connection to a server using a {@link StreamConnection} to process data.
*/
public class NioClient implements MessageWriteTarget {
private static final Logger log = LoggerFactory.getLogger(NioClient.class);
private final Handler handler;
private final NioClientManager manager = new NioClientManager();
class Handler implements TimeoutHandler, StreamConnection {
private final StreamConnection upstreamConnection;
private final SocketTimeoutTask timeoutTask;
private MessageWriteTarget writeTarget;
private boolean closeOnOpen = false;
private boolean closeCalled = false;
Handler(StreamConnection upstreamConnection, Duration connectTimeout) {
this.upstreamConnection = upstreamConnection;
this.timeoutTask = new SocketTimeoutTask(this::timeoutOccurred);
setSocketTimeout(connectTimeout);
setTimeoutEnabled(true);
}
private synchronized void timeoutOccurred() {
closeOnOpen = true;
connectionClosed();
}
@Override
public void setTimeoutEnabled(boolean timeoutEnabled) {
timeoutTask.setTimeoutEnabled(timeoutEnabled);
}
@Override
public void setSocketTimeout(Duration timeout) {
timeoutTask.setSocketTimeout(timeout);
}
@Override
public synchronized void connectionClosed() {
manager.stopAsync();
if (!closeCalled) {
closeCalled = true;
upstreamConnection.connectionClosed();
}
}
@Override
public synchronized void connectionOpened() {
if (!closeOnOpen)
upstreamConnection.connectionOpened();
}
@Override
public int receiveBytes(ByteBuffer buff) throws Exception {
return upstreamConnection.receiveBytes(buff);
}
@Override
public synchronized void setWriteTarget(MessageWriteTarget writeTarget) {
if (closeOnOpen)
writeTarget.closeConnection();
else {
setTimeoutEnabled(false);
this.writeTarget = writeTarget;
upstreamConnection.setWriteTarget(writeTarget);
}
}
@Override
public int getMaxMessageSize() {
return upstreamConnection.getMaxMessageSize();
}
}
/**
* Creates a new client to the given server address using the given {@link StreamConnection} to decode the data.
* The given connection MUST be unique to this object. This does not block while waiting for the connection to
* open, but will call either the {@link StreamConnection#connectionOpened()} or
* {@link StreamConnection#connectionClosed()} callback on the created network event processing thread.
* @param serverAddress socket address of the server to connect to
* @param parser parses data from the server
* @param connectTimeout timeout for establishing a connection to the server, or ZERO for no timeout
*/
public NioClient(final SocketAddress serverAddress, final StreamConnection parser,
final Duration connectTimeout) throws IOException {
manager.startAsync();
manager.awaitRunning();
handler = new Handler(parser, connectTimeout);
manager.openConnection(serverAddress, handler).whenComplete((result, t) -> {
if (t != null) {
log.error("Connect to {} failed: {}", serverAddress, Throwables.getRootCause(t));
}
});
}
@Override
public void closeConnection() {
handler.writeTarget.closeConnection();
}
@Override
public synchronized ListenableCompletableFuture writeBytes(byte[] message) throws IOException {
return handler.writeTarget.writeBytes(message);
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy