mirror of
https://github.com/moparisthebest/curl
synced 2024-12-25 09:38:54 -05:00
ssl: read pending close notify alert before closing the connection
This avoids a TCP reset (RST) if the server initiates a connection shutdown by sending an SSL close notify alert and then closes the TCP connection. For SSL connections, usually the server announces that it will close the connection with an SSL close notify alert. curl should read this alert. If curl does not read this alert and just closes the connection, some operating systems close the TCP connection with an RST flag. See RFC 1122, section 4.2.2.13 If curl reads the close notify alert, the TCP connection is closed normally with a FIN flag. The new code is similar to existing code in the "SSL shutdown" function: try to read an alert (non-blocking), and ignore any read errors. Closes #7095
This commit is contained in:
parent
8cc1fee5b9
commit
b249592d29
@ -1438,6 +1438,10 @@ static void close_one(struct ssl_connect_data *connssl)
|
||||
{
|
||||
struct ssl_backend_data *backend = connssl->backend;
|
||||
if(backend->session) {
|
||||
char buf[32];
|
||||
/* Maybe the server has already sent a close notify alert.
|
||||
Read it to avoid an RST on the TCP connection. */
|
||||
(void)gnutls_record_recv(backend->session, buf, sizeof(buf));
|
||||
gnutls_bye(backend->session, GNUTLS_SHUT_WR);
|
||||
gnutls_deinit(backend->session);
|
||||
backend->session = NULL;
|
||||
|
@ -813,8 +813,13 @@ static void mbedtls_close(struct Curl_easy *data,
|
||||
{
|
||||
struct ssl_connect_data *connssl = &conn->ssl[sockindex];
|
||||
struct ssl_backend_data *backend = connssl->backend;
|
||||
|
||||
char buf[32];
|
||||
(void) data;
|
||||
|
||||
/* Maybe the server has already sent a close notify alert.
|
||||
Read it to avoid an RST on the TCP connection. */
|
||||
(void)mbedtls_ssl_read(&backend->ssl, (unsigned char *)buf, sizeof(buf));
|
||||
|
||||
mbedtls_pk_free(&backend->pk);
|
||||
mbedtls_x509_crt_free(&backend->clicert);
|
||||
mbedtls_x509_crt_free(&backend->cacert);
|
||||
|
@ -1546,6 +1546,14 @@ static void close_one(struct ssl_connect_data *connssl)
|
||||
const bool client_cert = (backend->client_nickname != NULL)
|
||||
|| (backend->obj_clicert != NULL);
|
||||
|
||||
if(backend->handle) {
|
||||
char buf[32];
|
||||
/* Maybe the server has already sent a close notify alert.
|
||||
Read it to avoid an RST on the TCP connection. */
|
||||
(void)PR_Recv(backend->handle, buf, (int)sizeof(buf), 0,
|
||||
PR_INTERVAL_NO_WAIT);
|
||||
}
|
||||
|
||||
free(backend->client_nickname);
|
||||
backend->client_nickname = NULL;
|
||||
|
||||
|
@ -1400,7 +1400,13 @@ static void ossl_closeone(struct Curl_easy *data,
|
||||
{
|
||||
struct ssl_backend_data *backend = connssl->backend;
|
||||
if(backend->handle) {
|
||||
char buf[32];
|
||||
set_logger(conn, data);
|
||||
|
||||
/* Maybe the server has already sent a close notify alert.
|
||||
Read it to avoid an RST on the TCP connection. */
|
||||
(void)SSL_read(backend->handle, buf, (int)sizeof(buf));
|
||||
|
||||
(void)SSL_shutdown(backend->handle);
|
||||
SSL_set_connect_state(backend->handle);
|
||||
|
||||
|
@ -810,6 +810,10 @@ static void wolfssl_close(struct Curl_easy *data, struct connectdata *conn,
|
||||
(void) data;
|
||||
|
||||
if(backend->handle) {
|
||||
char buf[32];
|
||||
/* Maybe the server has already sent a close notify alert.
|
||||
Read it to avoid an RST on the TCP connection. */
|
||||
(void)SSL_read(backend->handle, buf, (int)sizeof(buf));
|
||||
(void)SSL_shutdown(backend->handle);
|
||||
SSL_free(backend->handle);
|
||||
backend->handle = NULL;
|
||||
|
Loading…
Reference in New Issue
Block a user