Additional fix for potential race condition which occurs when a user cancels a connection attempt

- Fixes #1986
This commit is contained in:
Trustin Lee 2013-11-18 17:00:23 +09:00
parent 54d3c99469
commit fd776274c9

View File

@ -167,10 +167,7 @@ public abstract class AbstractNioChannel extends AbstractChannel {
boolean wasActive = isActive(); boolean wasActive = isActive();
if (doConnect(remoteAddress, localAddress)) { if (doConnect(remoteAddress, localAddress)) {
promise.setSuccess(); fulfillConnectPromise(promise, wasActive);
if (!wasActive && isActive()) {
pipeline().fireChannelActive();
}
} else { } else {
connectPromise = promise; connectPromise = promise;
requestedRemoteAddress = remoteAddress; requestedRemoteAddress = remoteAddress;
@ -215,6 +212,22 @@ public abstract class AbstractNioChannel extends AbstractChannel {
} }
} }
private void fulfillConnectPromise(ChannelPromise promise, boolean wasActive) {
// trySuccess() will return false if a user cancelled the connection attempt.
boolean promiseSet = promise.trySuccess();
// Regardless if the connection attempt was cancelled, channelActive() event should be triggered,
// because what happened is what happened.
if (!wasActive && isActive()) {
pipeline().fireChannelActive();
}
// If a user cancelled the connection attempt, close the channel, which is followed by channelInactive().
if (!promiseSet) {
close(voidPromise());
}
}
@Override @Override
public void finishConnect() { public void finishConnect() {
// Note this method is invoked by the event loop only if the connection attempt was // Note this method is invoked by the event loop only if the connection attempt was
@ -226,10 +239,7 @@ public abstract class AbstractNioChannel extends AbstractChannel {
try { try {
boolean wasActive = isActive(); boolean wasActive = isActive();
doFinishConnect(); doFinishConnect();
connectPromise.setSuccess(); fulfillConnectPromise(connectPromise, wasActive);
if (!wasActive && isActive()) {
pipeline().fireChannelActive();
}
} catch (Throwable t) { } catch (Throwable t) {
if (t instanceof ConnectException) { if (t instanceof ConnectException) {
Throwable newT = new ConnectException(t.getMessage() + ": " + requestedRemoteAddress); Throwable newT = new ConnectException(t.getMessage() + ": " + requestedRemoteAddress);