From 54a481166a377ab16ab952267f255fab2e09131e Mon Sep 17 00:00:00 2001 From: Paul Lietar Date: Thu, 5 Sep 2024 19:01:24 +0100 Subject: [PATCH] Handle error responses without any content type. In some circumstances, Packit returns an HTTP error with no body and no Content-Type header. In those cases, httr2 returns NA as the content type. #166 added a check to make sure the content type is JSON before decoding the error, however the check did not handle NA values properly and gave a `missing value where TRUE/FALSE needed`. This now uses `identical` instead of `==` to compare the content type string against the expected string. --- DESCRIPTION | 2 +- R/outpack_http_client.R | 2 +- tests/testthat/helper-outpack-http.R | 4 +++- tests/testthat/test-outpack-http-client.R | 12 ++++++++++++ 4 files changed, 17 insertions(+), 3 deletions(-) diff --git a/DESCRIPTION b/DESCRIPTION index 1ddd209d..73d63ee6 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -1,6 +1,6 @@ Package: orderly2 Title: Orderly Next Generation -Version: 1.99.33 +Version: 1.99.34 Authors@R: c(person("Rich", "FitzJohn", role = c("aut", "cre"), email = "rich.fitzjohn@gmail.com"), person("Robert", "Ashton", role = "aut"), diff --git a/R/outpack_http_client.R b/R/outpack_http_client.R index 893379fc..991dc3f2 100644 --- a/R/outpack_http_client.R +++ b/R/outpack_http_client.R @@ -69,7 +69,7 @@ http_client_handle_error <- function(response) { ## for that and then reauthenticate; that requires that a callback ## is passed through here too. if (httr2::resp_is_error(response)) { - if (httr2::resp_content_type(response) == "application/json") { + if (identical(httr2::resp_content_type(response), "application/json")) { res <- httr2::resp_body_json(response) ## I am seeing Packit returning an element 'error' not a list of ## errors diff --git a/tests/testthat/helper-outpack-http.R b/tests/testthat/helper-outpack-http.R index 46438b93..12c9c857 100644 --- a/tests/testthat/helper-outpack-http.R +++ b/tests/testthat/helper-outpack-http.R @@ -4,7 +4,6 @@ mock_headers <- function(...) { mock_response <- function(content, status = 200L, wrap = TRUE, download = NULL) { - headers <- mock_headers() if (is.raw(content)) { headers <- mock_headers("content-type" = "application/octet-stream") } else if (inherits(content, "json")) { @@ -18,6 +17,9 @@ mock_response <- function(content, status = 200L, wrap = TRUE, } else if (is.character(content)) { headers <- mock_headers("content-type" = "text/plain") content <- c(writeBin(content, raw()), as.raw(0L)) + } else if (is.na(content)) { + headers <- mock_headers() + content <- raw(0) } else { stop("Unhandled mock response type") } diff --git a/tests/testthat/test-outpack-http-client.R b/tests/testthat/test-outpack-http-client.R index d793a14f..58bcee86 100644 --- a/tests/testthat/test-outpack-http-client.R +++ b/tests/testthat/test-outpack-http-client.R @@ -117,6 +117,18 @@ test_that("handle plain text errors", { }) +test_that("handle empty errors", { + local_mock_response(NA, status = 503L, wrap = FALSE) + + cl <- outpack_http_client$new("http://example.com", NULL) + err <- expect_error(cl$request("path"), "Service Unavailable") + + expect_s3_class(err, "outpack_http_client_error") + expect_equal(err$code, 503) + expect_null(err$errors) +}) + + test_that("can use the client to make requests", { skip_if_not_installed("mockery")