Correctly use HandshakeStatus.NEED_WRAP when a handshake failed and a alert was produced (#11412)

Motivation:

We need to ensure we always "consumed" all alerts etc via SSLEngine.wrap(...) before we teardown the engine. Failing to do so may lead to a situation where the remote peer will not be able to see the actual cause of the handshake failure but just see the connection being closed.

Modifications:

Correctly return HandshakeStatus.NEED_WRAP when we need to wrap some data first before we shutdown the engine because of a handshake failure.

Result:

Fixes https://github.com/netty/netty/issues/11388
This commit is contained in:
Norman Maurer 2021-06-24 10:05:47 +02:00
parent 4c8566f79b
commit 0a3ffc59e3

View File

@ -1312,8 +1312,7 @@ public class ReferenceCountedOpenSslEngine extends SSLEngine implements Referenc
}
}
private SSLEngineResult sslReadErrorResult(int error, int stackError, int bytesConsumed, int bytesProduced)
throws SSLException {
private boolean needWrapAgain(int stackError) {
// Check if we have a pending handshakeException and if so see if we need to consume all pending data from the
// BIO first or can just shutdown and throw it now.
// This is needed so we ensure close_notify etc is correctly send to the remote peer.
@ -1325,13 +1324,23 @@ public class ReferenceCountedOpenSslEngine extends SSLEngine implements Referenc
SSLException exception = handshakeState == HandshakeState.FINISHED ?
new SSLException(message) : new SSLHandshakeException(message);
if (pendingException == null) {
pendingException = exception;
pendingException = exception;
} else {
pendingException.addSuppressed(exception);
}
// We need to clear all errors so we not pick up anything that was left on the stack on the next
// operation. Note that shutdownWithError(...) will cleanup the stack as well so its only needed here.
SSL.clearError();
return true;
}
return false;
}
private SSLEngineResult sslReadErrorResult(int error, int stackError, int bytesConsumed, int bytesProduced)
throws SSLException {
if (needWrapAgain(stackError)) {
// There is something that needs to be send to the remote peer before we can teardown.
// This is most likely some alert.
return new SSLEngineResult(OK, NEED_WRAP, bytesConsumed, bytesProduced);
}
throw shutdownWithError("SSL_read", error, stackError);
@ -1899,6 +1908,11 @@ public class ReferenceCountedOpenSslEngine extends SSLEngine implements Referenc
return NEED_TASK;
}
if (needWrapAgain(SSL.getLastErrorNumber())) {
// There is something that needs to be send to the remote peer before we can teardown.
// This is most likely some alert.
return NEED_WRAP;
}
// Check if we have a pending exception that was created during the handshake and if so throw it after
// shutdown the connection.
if (pendingException != null) {