/* * Copyright (C) 2024-present ScyllaDB */ /* * SPDX-License-Identifier: LicenseRef-ScyllaDB-Source-Available-1.0 */ #define BOOST_TEST_MODULE object_storage #include "utils/s3/aws_error.hh" #include #include #include enum class message_style : uint8_t { singular = 1, plural = 2 }; namespace aws { std::ostream& boost_test_print_type(std::ostream& os, const aws::aws_error_type& error_type) { return os << fmt::underlying(error_type); } } // namespace aws static seastar::sstring build_xml_response(const std::string& exception, const std::string& message, const std::string& requestId, message_style style = message_style::singular) { return fmt::format(R"( {} {} {} {} /mybucket/myfile.bin {} {} {} {})", style == message_style::plural ? "" : "", style == message_style::plural ? "" : "", exception, message, style == message_style::singular ? "" + requestId + "" : "", style == message_style::plural ? "" : "", style == message_style::plural ? "" + requestId + "" : "", style == message_style::plural ? "" : ""); } BOOST_AUTO_TEST_CASE(TestXmlErrorPayload) { std::string message = "Test Message"; std::string requestId = "Request Id"; auto error = aws::aws_error::parse(build_xml_response("IncompleteSignatureException", message, requestId)).value(); BOOST_REQUIRE_EQUAL(aws::aws_error_type::INCOMPLETE_SIGNATURE, error.get_error_type()); BOOST_REQUIRE_EQUAL(message, error.get_error_message()); BOOST_REQUIRE_EQUAL(error.is_retryable(), utils::http::retryable::no); error = aws::aws_error::parse(build_xml_response("InternalFailure", message, requestId, message_style::plural)).value(); BOOST_REQUIRE_EQUAL(aws::aws_error_type::INTERNAL_FAILURE, error.get_error_type()); BOOST_REQUIRE_EQUAL(message, error.get_error_message()); BOOST_REQUIRE_EQUAL(error.is_retryable(), utils::http::retryable::yes); error = aws::aws_error::parse(build_xml_response("IDontExist", message, requestId, message_style::plural)).value(); BOOST_REQUIRE_EQUAL(aws::aws_error_type::UNKNOWN, error.get_error_type()); BOOST_REQUIRE_EQUAL(message, error.get_error_message()); BOOST_REQUIRE_EQUAL(error.is_retryable(), utils::http::retryable::no); auto no_error = aws::aws_error::parse(""); BOOST_REQUIRE_EQUAL(no_error.has_value(), false); no_error = aws::aws_error::parse("Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua."); BOOST_REQUIRE_EQUAL(no_error.has_value(), false); std::string response = " "; response += build_xml_response("InternalFailure", message, requestId, message_style::singular); error = aws::aws_error::parse(response).value(); BOOST_REQUIRE_EQUAL(aws::aws_error_type::INTERNAL_FAILURE, error.get_error_type()); BOOST_REQUIRE_EQUAL(message, error.get_error_message()); BOOST_REQUIRE_EQUAL(error.is_retryable(), utils::http::retryable::yes); } BOOST_AUTO_TEST_CASE(TestErrorsWithPrefixParse) { std::string message = "Test Message"; std::string exceptionPrefix = "blahblahblah#"; std::string requestId = "Request Id"; for (const auto& [exception, err] : aws::aws_error::get_errors()) { auto error = aws::aws_error::parse(build_xml_response(exceptionPrefix + std::string(exception), message, requestId)).value(); BOOST_REQUIRE_EQUAL(err.get_error_type(), error.get_error_type()); BOOST_REQUIRE_EQUAL(message, error.get_error_message()); BOOST_REQUIRE_EQUAL(err.is_retryable(), error.is_retryable()); } auto error = aws::aws_error::parse(build_xml_response(exceptionPrefix + "IDon'tExist", "JunkMessage", requestId)).value(); BOOST_REQUIRE_EQUAL(aws::aws_error_type::UNKNOWN, error.get_error_type()); BOOST_REQUIRE_EQUAL("JunkMessage", error.get_error_message()); BOOST_REQUIRE_EQUAL(error.is_retryable(), utils::http::retryable::no); } BOOST_AUTO_TEST_CASE(TestErrorsWithoutPrefixParse) { std::string message = "Test Message"; std::string requestId = "Request Id"; for (const auto& [exception, err] : aws::aws_error::get_errors()) { auto error = aws::aws_error::parse(build_xml_response(std::string(exception), message, requestId)).value(); BOOST_REQUIRE_EQUAL(err.get_error_type(), error.get_error_type()); BOOST_REQUIRE_EQUAL(message, error.get_error_message()); BOOST_REQUIRE_EQUAL(err.is_retryable(), error.is_retryable()); } auto error = aws::aws_error::parse(build_xml_response("IDon'tExist", "JunkMessage", requestId)).value(); BOOST_REQUIRE_EQUAL(aws::aws_error_type::UNKNOWN, error.get_error_type()); BOOST_REQUIRE_EQUAL("JunkMessage", error.get_error_message()); BOOST_REQUIRE_EQUAL(error.is_retryable(), utils::http::retryable::no); } BOOST_AUTO_TEST_CASE(TestHelperFunctions) { BOOST_REQUIRE_EQUAL(utils::http::from_http_code(seastar::http::reply::status_type::service_unavailable), utils::http::retryable::yes); BOOST_REQUIRE_EQUAL(utils::http::from_http_code(seastar::http::reply::status_type::unauthorized), utils::http::retryable::no); BOOST_REQUIRE_EQUAL(utils::http::from_system_error(std::system_error(ECONNRESET, std::system_category())), utils::http::retryable::yes); BOOST_REQUIRE_EQUAL(utils::http::from_system_error(std::system_error(EADDRINUSE, std::system_category())), utils::http::retryable::no); } BOOST_AUTO_TEST_CASE(TestNestedException) { // Test nested exceptions where the innermost is a system_error try { try { try { throw std::system_error(std::error_code(ECONNABORTED, std::system_category())); } catch (...) { std::throw_with_nested(std::runtime_error("Higher level runtime_error")); } } catch (...) { std::throw_with_nested(std::logic_error("Higher level logic_error")); } } catch (...) { auto error = aws::aws_error::from_exception_ptr(std::current_exception()); BOOST_REQUIRE_EQUAL(aws::aws_error_type::NETWORK_CONNECTION, error.get_error_type()); BOOST_REQUIRE_EQUAL("Software caused connection abort", error.get_error_message()); BOOST_REQUIRE_EQUAL(error.is_retryable(), utils::http::retryable::yes); } // Test nested exceptions where the innermost is NOT a system_error try { try { throw std::logic_error("Something bad happened"); } catch (...) { std::throw_with_nested(std::runtime_error("Higher level runtime_error")); } } catch (...) { auto error = aws::aws_error::from_exception_ptr(std::current_exception()); BOOST_REQUIRE_EQUAL(aws::aws_error_type::UNKNOWN, error.get_error_type()); BOOST_REQUIRE_EQUAL("Higher level runtime_error", error.get_error_message()); BOOST_REQUIRE_EQUAL(error.is_retryable(), utils::http::retryable::no); } // Test single exception which is NOT a nested exception try { throw std::runtime_error("Something bad happened"); } catch (...) { auto error = aws::aws_error::from_exception_ptr(std::current_exception()); BOOST_REQUIRE_EQUAL(aws::aws_error_type::UNKNOWN, error.get_error_type()); BOOST_REQUIRE_EQUAL("Something bad happened", error.get_error_message()); BOOST_REQUIRE_EQUAL(error.is_retryable(), utils::http::retryable::no); } // Test with non-std::exception try { throw "foo"; } catch (...) { auto error = aws::aws_error::from_exception_ptr(std::current_exception()); BOOST_REQUIRE_EQUAL(aws::aws_error_type::UNKNOWN, error.get_error_type()); BOOST_REQUIRE_EQUAL("No error message was provided, exception content: char const*", error.get_error_message()); BOOST_REQUIRE_EQUAL(error.is_retryable(), utils::http::retryable::no); } // Test system_error try { throw std::system_error(std::error_code(ECONNABORTED, std::system_category())); } catch (...) { auto error = aws::aws_error::from_exception_ptr(std::current_exception()); BOOST_REQUIRE_EQUAL(aws::aws_error_type::NETWORK_CONNECTION, error.get_error_type()); BOOST_REQUIRE_EQUAL("Software caused connection abort", error.get_error_message()); BOOST_REQUIRE_EQUAL(error.is_retryable(), utils::http::retryable::yes); } // Test aws_exception try { throw aws::aws_exception(aws::aws_error::get_errors().at("HTTP_TOO_MANY_REQUESTS")); } catch (...) { auto error = aws::aws_error::from_exception_ptr(std::current_exception()); BOOST_REQUIRE_EQUAL(aws::aws_error_type::HTTP_TOO_MANY_REQUESTS, error.get_error_type()); BOOST_REQUIRE_EQUAL("", error.get_error_message()); BOOST_REQUIRE_EQUAL(error.is_retryable(), utils::http::retryable::yes); } // Test httpd::unexpected_status_error try { throw seastar::httpd::unexpected_status_error(seastar::http::reply::status_type::network_connect_timeout); } catch (...) { auto error = aws::aws_error::from_exception_ptr(std::current_exception()); BOOST_REQUIRE_EQUAL(aws::aws_error_type::HTTP_NETWORK_CONNECT_TIMEOUT, error.get_error_type()); BOOST_REQUIRE_EQUAL(" HTTP code: 599 Network Connect Timeout", error.get_error_message()); BOOST_REQUIRE_EQUAL(error.is_retryable(), utils::http::retryable::yes); } }