Skip to content

Commit

Permalink
Update from httr to httr2.
Browse files Browse the repository at this point in the history
httr2 has some nice functions for OAuth2 support, which we'll want for
mrc-5725.

Most of the changes come from a different way of customizing requests,
which transpires to the location_http implementation, and a different
way of setting up mocks (which is arguably nicer).
  • Loading branch information
plietar committed Aug 30, 2024
1 parent 6977728 commit de9c38c
Show file tree
Hide file tree
Showing 8 changed files with 203 additions and 134 deletions.
4 changes: 2 additions & 2 deletions DESCRIPTION
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
Package: orderly2
Title: Orderly Next Generation
Version: 1.99.27
Version: 1.99.28
Authors@R: c(person("Rich", "FitzJohn", role = c("aut", "cre"),
email = "[email protected]"),
person("Robert", "Ashton", role = "aut"),
Expand Down Expand Up @@ -28,7 +28,7 @@ Imports:
Suggests:
DBI,
RSQLite,
httr,
httr2,
jsonvalidate (>= 1.4.0),
knitr,
mockery,
Expand Down
34 changes: 19 additions & 15 deletions R/location_http.R
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ orderly_location_http <- R6::R6Class(
},

list = function() {
dat <- private$client$get("/metadata/list", parse_json = TRUE)$data
dat <- private$client$request("/metadata/list")$data
data_frame(
packet = vcapply(dat, "[[", "packet"),
time = num_to_time(vnapply(dat, "[[", "time")),
Expand All @@ -21,8 +21,8 @@ orderly_location_http <- R6::R6Class(
metadata = function(packet_ids) {
ret <- vcapply(packet_ids, function(id) {
tryCatch(
trimws(private$client$get(sprintf("/metadata/%s/text", id),
parse_json = FALSE)),
trimws(private$client$request(sprintf("/metadata/%s/text", id),
parse_json = FALSE)),
outpack_http_client_error = function(e) {
if (e$code == 404) {
e$message <- sprintf("Some packet ids not found: '%s'", id)
Expand All @@ -43,7 +43,7 @@ orderly_location_http <- R6::R6Class(
## progress in the client, but there's not much point until
## then.
tryCatch(
private$client$get(sprintf("/file/%s", hash), download = dest),
private$client$request(sprintf("/file/%s", hash), download = dest),
outpack_http_client_error = function(e) {
if (e$code == 404) {
unlink(dest)
Expand All @@ -56,32 +56,36 @@ orderly_location_http <- R6::R6Class(

## TODO: we could get the schemas here from outpack_server too
list_unknown_packets = function(ids) {
body <- to_json(list(ids = ids, unpacked = scalar(TRUE)), NULL)
content <- httr::content_type("application/json")
res <- private$client$post("/packets/missing", body, content)
res <- private$client$request(
"/packets/missing",
function(r) r |> httr2::req_body_json(list(ids = ids, unpacked = scalar(TRUE))))
list_to_character(res$data)
},

list_unknown_files = function(hashes) {
body <- to_json(list(hashes = hashes), NULL)
res <- private$client$post("/files/missing", body,
httr::content_type("application/json"))
res <- private$client$request(
"/files/missing",
function(r) r |> httr2::req_body_json(list(hashes = hashes)))
list_to_character(res$data)
},

push_file = function(src, hash) {
body <- httr::upload_file(src, "application/octet-stream")
res <- private$client$post(sprintf("/file/%s", hash), body)
res <- private$client$request(
sprintf("/file/%s", hash),
function(r) r |> httr2::req_body_file(src, "application/octet-stream"))

invisible(NULL)
},

push_metadata = function(packet_id, hash, path) {
meta <- read_string(path)
res <- private$client$post(sprintf("/packet/%s", hash), meta,
httr::content_type("text/plain"))
res <- private$client$request(
sprintf("/packet/%s", hash),
function(r) r |> httr2::req_body_raw(meta, "text/plain"))
invisible(NULL)
}
))
)
)


orderly_location_packit <- function(url, token) {
Expand Down
73 changes: 35 additions & 38 deletions R/outpack_http_client.R
Original file line number Diff line number Diff line change
Expand Up @@ -21,36 +21,37 @@ outpack_http_client <- R6::R6Class(
}
},

get = function(path, ...) {
request = function(path, customize = identity, ...) {
self$authorise()
http_client_request(httr::GET, paste0(self$url, path), ...,
self$auth$header)
},

post = function(path, body, ...) {
self$authorise()
http_client_request(httr::POST, paste0(self$url, path), body = body, ...,
self$auth$header)
http_client_request(
self$url,
function(r) {
r |>
httr2::req_url_path_append(path) |>
httr2::req_headers(!!!self$auth$header) |>
customize()
}, ...)
}
))


http_client_request <- function(verb, url, ..., parse_json = TRUE,
download = NULL) {
if (is.null(download)) {
response <- verb(url, ...)
} else {
response <- verb(url, ...,
http_client_download_options(download))
http_client_request <- function(url, customize = identity, download = NULL, parse_json = TRUE) {
req <- httr2::request(url)
if (!is.null(download)) {
req <- httr2::req_headers(req, Accept = "application/octet-stream")
}

http_client_handle_error(response)
req <- customize(req)
resp <- tryCatch(
httr2::req_perform(req, path = download),
httr2_http = function(cnd) {
http_client_handle_error(cnd$resp)
})

if (is.null(download)) {
txt <- httr::content(response, "text", encoding = "UTF-8")
if (parse_json) {
from_json(txt)
httr2::resp_body_json(resp, simplifyVector = FALSE)
} else {
txt
httr2::resp_body_string(resp)
}
} else {
download
Expand All @@ -63,17 +64,19 @@ http_client_handle_error <- function(response) {
## that an expired timeout produces a certain error code we watch
## for that and then reauthenticate; that requires that a callback
## is passed through here too.
code <- httr::status_code(response)
if (httr::http_error(code)) {
if (httr::http_type(response) == "application/json") {
txt <- httr::content(response, "text", encoding = "UTF-8")
res <- from_json(txt)
if (httr2::resp_is_error(response)) {
if (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
errors <- if ("error" %in% names(res)) list(res$error) else res$errors
stop(http_client_error(errors[[1]]$detail, code, errors))
stop(http_client_error(errors[[1]]$detail,
httr2::resp_status(response),
errors))
} else {
stop(http_client_error(httr::http_status(code)$message, code, NULL))
stop(http_client_error(httr2::resp_status_desc(response),
httr2::resp_status(response),
NULL))
}
}
response
Expand All @@ -87,12 +90,6 @@ http_client_error <- function(msg, code, errors) {
}


http_client_download_options <- function(dest) {
c(httr::write_disk(dest),
httr::accept("application/octet-stream"))
}


## Logging in with packit is quite slow and we'll want to cache this;
## but we won't be holding a persistant handle to the root. So for
## now at least we'll keep a pool of generated bearer token headers,
Expand All @@ -108,11 +105,11 @@ http_client_login <- function(name, auth) {
key <- rlang::hash(auth)
if (is.null(auth_cache[[key]])) {
cli::cli_alert_info("Logging in to {name}")
res <- http_client_request(httr::POST, auth$url,
body = auth$data, encode = "json")

res <- http_client_request(auth$url, . %>% httr2::req_body_json(auth$data))

cli::cli_alert_success("Logged in successfully")
auth_cache[[key]] <- httr::add_headers(
"Authorization" = paste("Bearer", res$token))
auth_cache[[key]] <- list("Authorization" = paste("Bearer", res$token))
}
auth_cache[[key]]
}
5 changes: 0 additions & 5 deletions R/util.R
Original file line number Diff line number Diff line change
Expand Up @@ -332,11 +332,6 @@ to_json <- function(x, schema = NULL, auto_unbox = FALSE, ...) {
}


from_json <- function(x, ...) {
jsonlite::fromJSON(x, simplifyVector = FALSE, ...)
}


as_json <- function(str) {
assert_scalar_character(str)
structure(str, class = "json")
Expand Down
21 changes: 15 additions & 6 deletions tests/testthat/helper-outpack-http.R
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,8 @@ mock_headers <- function(...) {
mock_response <- function(content, status = 200L, wrap = TRUE,
download = NULL) {
headers <- mock_headers()
if (!is.null(download)) {
if (inherits(content, "raw")) {
headers <- mock_headers("content-type" = "application/octet-stream")
writeBin(content, download)
} else if (inherits(content, "json")) {
headers <- mock_headers("content-type" = "application/json")
if (wrap) {
Expand All @@ -22,10 +21,14 @@ mock_response <- function(content, status = 200L, wrap = TRUE,
} else {
stop("Unhandled mock response type")
}
structure(list(status_code = status,
headers = headers,
content = content),
class = "response")

if (!is.null(download)) {
writeBin(content, download)
}

httr2::response(status_code = status,
headers = headers,
body = content)
}


Expand All @@ -38,3 +41,9 @@ json_string <- function(s) {
clear_auth_cache <- function() {
rm(list = ls(auth_cache), envir = auth_cache)
}

local_mock_response <- function(..., env = rlang::caller_env()) {
mock <- mockery::mock(mock_response(...))
httr2::local_mocked_responses(function(req) mock(req), env = env)
mock
}
4 changes: 3 additions & 1 deletion tests/testthat/helper-outpack-server.R
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,9 @@ outpack_server <- function(path, timeout = 10) {
t_end <- Sys.time() + timeout
success <- FALSE
while (!success && px$is_alive() && Sys.time() < t_end) {
result <- tryCatch(httr::GET("http://localhost:8000"), error = identity)
result <- tryCatch(
httr2::request("http://localhost:8000") |> httr2::req_perform(),
error = identity)
success <- !inherits(result, "error")
}
if (!success) {
Expand Down
Loading

0 comments on commit de9c38c

Please sign in to comment.