From ccc2260117f6aec58bf3f2d1703ba1b374189fd2 Mon Sep 17 00:00:00 2001 From: Russell Cohen Date: Tue, 5 Sep 2023 15:55:43 -0400 Subject: [PATCH] Retry additional classes of H2 errors This PR adds two additional classes of retries tested: 1. GO_AWAY: https://github.com/awslabs/aws-sdk-rust/issues/738 2. REFUSED_STREAM: https://github.com/awslabs/aws-sdk-rust/issues/858 I tested 1 by using the example helpfully provided. The fix for 2 is untested and difficult to reproduce but since my fix for 1 worked, I'm confident that we're detecting the correct error class here. --- rust-runtime/aws-smithy-client/Cargo.toml | 3 +- .../aws-smithy-client/src/hyper_ext.rs | 34 +++++++++++++------ 2 files changed, 25 insertions(+), 12 deletions(-) diff --git a/rust-runtime/aws-smithy-client/Cargo.toml b/rust-runtime/aws-smithy-client/Cargo.toml index c691e851ef..8e8c5a39c7 100644 --- a/rust-runtime/aws-smithy-client/Cargo.toml +++ b/rust-runtime/aws-smithy-client/Cargo.toml @@ -14,7 +14,7 @@ wiremock = ["test-util", "dep:hyper", "hyper?/server", "hyper?/h2", "rustls", "t native-tls = [] allow-compilation = [] # our tests use `cargo test --all-features` and native-tls breaks CI rustls = ["dep:hyper-rustls", "dep:lazy_static", "dep:rustls", "client-hyper", "rt-tokio"] -client-hyper = ["dep:hyper", "hyper?/client", "hyper?/http2", "hyper?/http1", "hyper?/tcp"] +client-hyper = ["dep:hyper", "hyper?/client", "hyper?/http2", "hyper?/http1", "hyper?/tcp", "dep:h2"] hyper-webpki-doctest-only = ["dep:hyper-rustls", "hyper-rustls?/webpki-roots"] [dependencies] @@ -28,6 +28,7 @@ fastrand = "2.0.0" http = "0.2.3" http-body = "0.4.4" hyper = { version = "0.14.26", default-features = false, optional = true } +h2 = { version = "0.3", default-features = false, optional = true } # cargo does not support optional test dependencies, so to completely disable rustls # we need to add the webpki-roots feature here. # https://github.com/rust-lang/cargo/issues/1596 diff --git a/rust-runtime/aws-smithy-client/src/hyper_ext.rs b/rust-runtime/aws-smithy-client/src/hyper_ext.rs index be1e5679d2..6eee6a4f61 100644 --- a/rust-runtime/aws-smithy-client/src/hyper_ext.rs +++ b/rust-runtime/aws-smithy-client/src/hyper_ext.rs @@ -92,6 +92,7 @@ use hyper::client::connect::{ capture_connection, CaptureConnection, Connected, Connection, HttpInfo, }; +use h2::Reason; use std::error::Error; use std::fmt::Debug; @@ -196,20 +197,31 @@ fn downcast_error(err: BoxError) -> ConnectorError { /// Convert a [`hyper::Error`] into a [`ConnectorError`] fn to_connector_error(err: hyper::Error) -> ConnectorError { if err.is_timeout() || find_source::(&err).is_some() { - ConnectorError::timeout(err.into()) - } else if err.is_user() { - ConnectorError::user(err.into()) - } else if err.is_closed() || err.is_canceled() || find_source::(&err).is_some() - { - ConnectorError::io(err.into()) + return ConnectorError::timeout(err.into()); + } + if err.is_user() { + return ConnectorError::user(err.into()); + } + if err.is_closed() || err.is_canceled() || find_source::(&err).is_some() { + return ConnectorError::io(err.into()); } // We sometimes receive this from S3: hyper::Error(IncompleteMessage) - else if err.is_incomplete_message() { - ConnectorError::other(err.into(), Some(ErrorKind::TransientError)) - } else { - tracing::warn!(err = %DisplayErrorContext(&err), "unrecognized error from Hyper. If this error should be retried, please file an issue."); - ConnectorError::other(err.into(), None) + if err.is_incomplete_message() { + return ConnectorError::other(err.into(), Some(ErrorKind::TransientError)); } + if err.is_closed() { + return ConnectorError::io(err.into()); + } + if let Some(h2_err) = find_source::(&err) { + if h2_err.is_go_away() + || (h2_err.is_reset() && h2_err.reason() == Some(Reason::REFUSED_STREAM)) + { + return ConnectorError::io(err.into()); + } + } + + tracing::warn!(err = %DisplayErrorContext(&err), "unrecognized error from Hyper. If this error should be retried, please file an issue."); + ConnectorError::other(err.into(), None) } fn find_source<'a, E: Error + 'static>(err: &'a (dyn Error + 'static)) -> Option<&'a E> {