mirror of
https://github.com/haveno-dex/haveno.git
synced 2025-01-18 08:45:02 +00:00
update ProtoOutputStream from upstream
This commit is contained in:
parent
e7371d1299
commit
20b55ed9dd
3 changed files with 75 additions and 100 deletions
|
@ -170,11 +170,11 @@ public class Connection implements HasCapabilities, Runnable, MessageListener {
|
|||
///////////////////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
Connection(Socket socket,
|
||||
MessageListener messageListener,
|
||||
ConnectionListener connectionListener,
|
||||
@Nullable NodeAddress peersNodeAddress,
|
||||
NetworkProtoResolver networkProtoResolver,
|
||||
@Nullable BanFilter banFilter) {
|
||||
MessageListener messageListener,
|
||||
ConnectionListener connectionListener,
|
||||
@Nullable NodeAddress peersNodeAddress,
|
||||
NetworkProtoResolver networkProtoResolver,
|
||||
@Nullable BanFilter banFilter) {
|
||||
this.socket = socket;
|
||||
this.connectionListener = connectionListener;
|
||||
this.banFilter = banFilter;
|
||||
|
@ -200,7 +200,7 @@ public class Connection implements HasCapabilities, Runnable, MessageListener {
|
|||
// When you construct an ObjectInputStream, in the constructor the class attempts to read a header that
|
||||
// the associated ObjectOutputStream on the other end of the connection has written.
|
||||
// It will not return until that header has been read.
|
||||
protoOutputStream = new SynchronizedProtoOutputStream(socket.getOutputStream(), statistic);
|
||||
protoOutputStream = new ProtoOutputStream(socket.getOutputStream(), statistic);
|
||||
protoInputStream = socket.getInputStream();
|
||||
// We create a thread for handling inputStream data
|
||||
executorService.submit(this);
|
||||
|
@ -239,8 +239,7 @@ public class Connection implements HasCapabilities, Runnable, MessageListener {
|
|||
if (banFilter != null &&
|
||||
peersNodeAddressOptional.isPresent() &&
|
||||
banFilter.isPeerBanned(peersNodeAddressOptional.get())) {
|
||||
log.warn("We tried to send a message to a banned peer. message={}",
|
||||
networkEnvelope.getClass().getSimpleName());
|
||||
log.warn("We tried to send a message to a banned peer. message={}", networkEnvelope.getClass().getSimpleName());
|
||||
reportInvalidRequest(RuleViolation.PEER_BANNED);
|
||||
return;
|
||||
}
|
||||
|
@ -256,7 +255,7 @@ public class Connection implements HasCapabilities, Runnable, MessageListener {
|
|||
long elapsed = now - lastSendTimeStamp;
|
||||
if (elapsed < getSendMsgThrottleTrigger()) {
|
||||
log.debug("We got 2 sendMessage requests in less than {} ms. We set the thread to sleep " +
|
||||
"for {} ms to avoid flooding our peer. lastSendTimeStamp={}, now={}, elapsed={}, networkEnvelope={}",
|
||||
"for {} ms to avoid flooding our peer. lastSendTimeStamp={}, now={}, elapsed={}, networkEnvelope={}",
|
||||
getSendMsgThrottleTrigger(), getSendMsgThrottleSleep(), lastSendTimeStamp, now, elapsed,
|
||||
networkEnvelope.getClass().getSimpleName());
|
||||
|
||||
|
@ -437,6 +436,7 @@ public class Connection implements HasCapabilities, Runnable, MessageListener {
|
|||
messageListeners.forEach(listener -> listener.onMessage(envelope, connection))));
|
||||
}
|
||||
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////////////////
|
||||
// Setters
|
||||
///////////////////////////////////////////////////////////////////////////////////////////
|
||||
|
@ -456,6 +456,7 @@ public class Connection implements HasCapabilities, Runnable, MessageListener {
|
|||
peersNodeAddressProperty.set(peerNodeAddress);
|
||||
}
|
||||
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////////////////
|
||||
// Getters
|
||||
///////////////////////////////////////////////////////////////////////////////////////////
|
||||
|
@ -498,7 +499,8 @@ public class Connection implements HasCapabilities, Runnable, MessageListener {
|
|||
|
||||
Uninterruptibles.sleepUninterruptibly(200, TimeUnit.MILLISECONDS);
|
||||
} catch (Throwable t) {
|
||||
handleException(t);
|
||||
log.error(t.getMessage());
|
||||
t.printStackTrace();
|
||||
} finally {
|
||||
stopped = true;
|
||||
EXECUTOR.execute(() -> doShutDown(closeConnectionReason, shutDownCompleteHandler));
|
||||
|
@ -611,8 +613,7 @@ public class Connection implements HasCapabilities, Runnable, MessageListener {
|
|||
"connection with address{} and uid {}", ruleViolations, peersNodeAddressProperty, uid);
|
||||
this.ruleViolation = ruleViolation;
|
||||
if (ruleViolation == RuleViolation.PEER_BANNED) {
|
||||
log.debug("We close connection due RuleViolation.PEER_BANNED. peersNodeAddress={}",
|
||||
getPeersNodeAddressOptional());
|
||||
log.debug("We close connection due RuleViolation.PEER_BANNED. peersNodeAddress={}", getPeersNodeAddressOptional());
|
||||
shutDown(CloseConnectionReason.PEER_BANNED);
|
||||
} else if (ruleViolation == RuleViolation.INVALID_CLASS) {
|
||||
log.warn("We close connection due RuleViolation.INVALID_CLASS");
|
||||
|
@ -655,8 +656,8 @@ public class Connection implements HasCapabilities, Runnable, MessageListener {
|
|||
// TODO sometimes we get StreamCorruptedException, OptionalDataException, IllegalStateException
|
||||
closeConnectionReason = CloseConnectionReason.UNKNOWN_EXCEPTION;
|
||||
log.warn("Unknown reason for exception at socket: {}\n\t" +
|
||||
"peer={}\n\t" +
|
||||
"Exception={}",
|
||||
"peer={}\n\t" +
|
||||
"Exception={}",
|
||||
socket.toString(),
|
||||
this.peersNodeAddressOptional,
|
||||
e.toString());
|
||||
|
@ -756,7 +757,7 @@ public class Connection implements HasCapabilities, Runnable, MessageListener {
|
|||
long elapsed = now - lastReadTimeStamp;
|
||||
if (elapsed < 10) {
|
||||
log.debug("We got 2 network messages received in less than 10 ms. We set the thread to sleep " +
|
||||
"for 20 ms to avoid getting flooded by our peer. lastReadTimeStamp={}, now={}, elapsed={}",
|
||||
"for 20 ms to avoid getting flooded by our peer. lastReadTimeStamp={}, now={}, elapsed={}",
|
||||
lastReadTimeStamp, now, elapsed);
|
||||
Thread.sleep(20);
|
||||
}
|
||||
|
@ -803,7 +804,7 @@ public class Connection implements HasCapabilities, Runnable, MessageListener {
|
|||
if (!proto.getMessageVersion().equals(Version.getP2PMessageVersion())
|
||||
&& reportInvalidRequest(RuleViolation.WRONG_NETWORK_ID)) {
|
||||
log.warn("RuleViolation.WRONG_NETWORK_ID. version of message={}, app version={}, " +
|
||||
"proto.toTruncatedString={}", proto.getMessageVersion(),
|
||||
"proto.toTruncatedString={}", proto.getMessageVersion(),
|
||||
Version.getP2PMessageVersion(),
|
||||
Utilities.toTruncatedString(proto.toString()));
|
||||
return;
|
||||
|
@ -821,8 +822,7 @@ public class Connection implements HasCapabilities, Runnable, MessageListener {
|
|||
|
||||
if (CloseConnectionReason.PEER_BANNED.name().equals(proto.getCloseConnectionMessage().getReason())) {
|
||||
log.warn("We got shut down because we are banned by the other peer. " +
|
||||
"(InputHandler.run CloseConnectionMessage). Peer: {}",
|
||||
getPeersNodeAddressOptional());
|
||||
"(InputHandler.run CloseConnectionMessage). Peer: {}", getPeersNodeAddressOptional());
|
||||
}
|
||||
shutDown(CloseConnectionReason.CLOSE_REQUESTED_BY_PEER);
|
||||
return;
|
||||
|
@ -841,8 +841,7 @@ public class Connection implements HasCapabilities, Runnable, MessageListener {
|
|||
}
|
||||
|
||||
if (!(networkEnvelope instanceof SendersNodeAddressMessage) && peersNodeAddressOptional.isEmpty()) {
|
||||
log.info("We got a {} from a peer with yet unknown address on connection with uid={}",
|
||||
networkEnvelope.getClass().getSimpleName(), uid);
|
||||
log.info("We got a {} from a peer with yet unknown address on connection with uid={}", networkEnvelope.getClass().getSimpleName(), uid);
|
||||
}
|
||||
|
||||
EXECUTOR.execute(() -> onMessage(networkEnvelope, this));
|
||||
|
|
|
@ -17,49 +17,84 @@
|
|||
|
||||
package haveno.network.p2p.network;
|
||||
|
||||
import haveno.common.proto.network.NetworkEnvelope;
|
||||
import haveno.network.p2p.peers.keepalive.messages.KeepAliveMessage;
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import javax.annotation.concurrent.NotThreadSafe;
|
||||
import haveno.common.proto.network.NetworkEnvelope;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.io.OutputStream;
|
||||
|
||||
@NotThreadSafe
|
||||
import java.util.concurrent.TimeUnit;
|
||||
import java.util.concurrent.atomic.AtomicBoolean;
|
||||
import java.util.concurrent.locks.Lock;
|
||||
import java.util.concurrent.locks.ReentrantLock;
|
||||
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import javax.annotation.concurrent.ThreadSafe;
|
||||
|
||||
@ThreadSafe
|
||||
class ProtoOutputStream {
|
||||
private static final Logger log = LoggerFactory.getLogger(ProtoOutputStream.class);
|
||||
|
||||
private final OutputStream delegate;
|
||||
private final OutputStream outputStream;
|
||||
private final Statistic statistic;
|
||||
|
||||
ProtoOutputStream(OutputStream delegate, Statistic statistic) {
|
||||
this.delegate = delegate;
|
||||
private final AtomicBoolean isConnectionActive = new AtomicBoolean(true);
|
||||
private final Lock lock = new ReentrantLock();
|
||||
|
||||
ProtoOutputStream(OutputStream outputStream, Statistic statistic) {
|
||||
this.outputStream = outputStream;
|
||||
this.statistic = statistic;
|
||||
}
|
||||
|
||||
void writeEnvelope(NetworkEnvelope envelope) {
|
||||
lock.lock();
|
||||
|
||||
try {
|
||||
writeEnvelopeOrThrow(envelope);
|
||||
} catch (IOException e) {
|
||||
if (!isConnectionActive.get()) {
|
||||
// Connection was closed by us.
|
||||
return;
|
||||
}
|
||||
|
||||
log.error("Failed to write envelope", e);
|
||||
throw new HavenoRuntimeException("Failed to write envelope", e);
|
||||
|
||||
} finally {
|
||||
lock.unlock();
|
||||
}
|
||||
}
|
||||
|
||||
void onConnectionShutdown() {
|
||||
isConnectionActive.set(false);
|
||||
|
||||
boolean acquiredLock = tryToAcquireLock();
|
||||
if (!acquiredLock) {
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
delegate.close();
|
||||
outputStream.close();
|
||||
} catch (Throwable t) {
|
||||
log.error("Failed to close connection", t);
|
||||
|
||||
} finally {
|
||||
lock.unlock();
|
||||
}
|
||||
}
|
||||
|
||||
private void writeEnvelopeOrThrow(NetworkEnvelope envelope) throws IOException {
|
||||
long ts = System.currentTimeMillis();
|
||||
protobuf.NetworkEnvelope proto = envelope.toProtoNetworkEnvelope();
|
||||
proto.writeDelimitedTo(delegate);
|
||||
delegate.flush();
|
||||
|
||||
proto.writeDelimitedTo(outputStream);
|
||||
outputStream.flush();
|
||||
long duration = System.currentTimeMillis() - ts;
|
||||
if (duration > 10000) {
|
||||
log.info("Sending {} to peer took {} sec.", envelope.getClass().getSimpleName(), duration / 1000d);
|
||||
}
|
||||
statistic.addSentBytes(proto.getSerializedSize());
|
||||
statistic.addSentMessage(envelope);
|
||||
|
||||
|
@ -67,4 +102,13 @@ class ProtoOutputStream {
|
|||
statistic.updateLastActivityTimestamp();
|
||||
}
|
||||
}
|
||||
|
||||
private boolean tryToAcquireLock() {
|
||||
long shutdownTimeout = Connection.getShutdownTimeout();
|
||||
try {
|
||||
return lock.tryLock(shutdownTimeout, TimeUnit.MILLISECONDS);
|
||||
} catch (InterruptedException e) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,68 +0,0 @@
|
|||
/*
|
||||
* This file is part of Haveno.
|
||||
*
|
||||
* Haveno is free software: you can redistribute it and/or modify it
|
||||
* under the terms of the GNU Affero General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or (at
|
||||
* your option) any later version.
|
||||
*
|
||||
* Haveno is distributed in the hope that it will be useful, but WITHOUT
|
||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
|
||||
* License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Affero General Public License
|
||||
* along with Haveno. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
package haveno.network.p2p.network;
|
||||
|
||||
import haveno.common.proto.network.NetworkEnvelope;
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import javax.annotation.concurrent.ThreadSafe;
|
||||
import java.io.OutputStream;
|
||||
import java.util.concurrent.ExecutionException;
|
||||
import java.util.concurrent.ExecutorService;
|
||||
import java.util.concurrent.Executors;
|
||||
import java.util.concurrent.Future;
|
||||
|
||||
@ThreadSafe
|
||||
class SynchronizedProtoOutputStream extends ProtoOutputStream {
|
||||
private static final Logger log = LoggerFactory.getLogger(SynchronizedProtoOutputStream.class);
|
||||
|
||||
private final ExecutorService executorService;
|
||||
|
||||
SynchronizedProtoOutputStream(OutputStream delegate, Statistic statistic) {
|
||||
super(delegate, statistic);
|
||||
this.executorService = Executors.newSingleThreadExecutor();
|
||||
}
|
||||
|
||||
@Override
|
||||
void writeEnvelope(NetworkEnvelope envelope) {
|
||||
Future<?> future = executorService.submit(() -> super.writeEnvelope(envelope));
|
||||
try {
|
||||
future.get();
|
||||
} catch (InterruptedException e) {
|
||||
Thread currentThread = Thread.currentThread();
|
||||
currentThread.interrupt();
|
||||
String msg = "Thread " + currentThread + " was interrupted. InterruptedException=" + e;
|
||||
log.error(msg);
|
||||
throw new HavenoRuntimeException(msg, e);
|
||||
} catch (ExecutionException e) {
|
||||
String msg = "Failed to write envelope. ExecutionException " + e;
|
||||
log.error(msg);
|
||||
throw new HavenoRuntimeException(msg, e);
|
||||
}
|
||||
}
|
||||
|
||||
void onConnectionShutdown() {
|
||||
try {
|
||||
executorService.shutdownNow();
|
||||
super.onConnectionShutdown();
|
||||
} catch (Throwable t) {
|
||||
log.error("Failed to handle connection shutdown. Throwable={}", t.toString());
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Reference in a new issue