#' Create a new HTTP response
#'
#' @description
#' Generally, you should not need to call this function directly; you'll
#' get a real HTTP response by calling [req_perform()] and friends. This
#' function is provided primarily for testing, and a place to describe
#' the key components of a response.
#'
#' `response()` creates a generic response; `response_json()` creates a
#' response with a JSON body, automatically adding the correct Content-Type
#' header.
#'
#' @keywords internal
#' @param status_code HTTP status code. Must be a single integer.
#' @param url URL response came from; might not be the same as the URL in
#' the request if there were any redirects.
#' @param method HTTP method used to retrieve the response.
#' @param headers HTTP headers. Can be supplied as a raw or character vector
#' which will be parsed using the standard rules, or a named list.
#' @param body Response, if any, contained in the response body.
#' For `response_json()`, a R data structure to serialize to JSON.
#' @returns An HTTP response: an S3 list with class `httr2_response`.
#' @export
#' @examples
#' response()
#' response(404, method = "POST")
#' response(headers = c("Content-Type: text/html", "Content-Length: 300"))
response <- function(status_code = 200,
url = "https://example.com",
method = "GET",
headers = list(),
body = raw()) {
check_number_whole(status_code, min = 100, max = 700)
check_string(url)
check_string(method)
headers <- as_headers(headers)
new_response(
method = method,
url = url,
status_code = as.integer(status_code),
headers = headers,
body = body
)
}
#' @export
#' @rdname response
response_json <- function(status_code = 200,
url = "https://example.com",
method = "GET",
headers = list(),
body = list()) {
headers <- as_headers(headers)
headers$`Content-Type` <- "application/json"
body <- charToRaw(jsonlite::toJSON(body, auto_unbox = TRUE))
new_response(
method = method,
url = url,
status_code = as.integer(status_code),
headers = headers,
body = body
)
}
new_response <- function(method,
url,
status_code,
headers,
body,
request = NULL,
error_call = caller_env()) {
check_string(method, call = error_call)
check_string(url, call = error_call)
check_number_whole(status_code, call = error_call)
check_request(request, allow_null = TRUE)
headers <- as_headers(headers, error_call = error_call)
# ensure we always have a date field
if (!"date" %in% tolower(names(headers))) {
headers$Date <- "Wed, 01 Jan 2020 00:00:00 UTC"
}
structure(
list(
method = method,
url = url,
status_code = status_code,
headers = headers,
body = body,
request = request,
cache = new_environment()
),
class = "httr2_response"
)
}
#' @export
print.httr2_response <- function(x, ...) {
cli::cli_text("{.cls {class(x)}}")
cli::cli_text("{.strong {x$method}} {x$url}")
cli::cli_text("{.field Status}: {x$status_code} {resp_status_desc(x)}")
if (resp_header_exists(x, "Content-Type")) {
cli::cli_text("{.field Content-Type}: {resp_content_type(x)}")
}
body <- x$body
if (!resp_has_body(x)) {
cli::cli_text("{.field Body}: None")
} else {
switch(resp_body_type(x),
disk = cli::cli_text("{.field Body}: On disk {.path {body}} ({file.size(body)} bytes)"),
memory = cli::cli_text("{.field Body}: In memory ({length(body)} bytes)"),
stream = cli::cli_text("{.field Body}: Streaming connection")
)
}
invisible(x)
}
#' Show the raw response
#'
#' This function reconstructs the HTTP message that httr2 received from the
#' server. It's unlikely to be exactly byte-for-byte identical (because most
#' servers compress at least the body, and HTTP/2 can also compress the
#' headers), but it conveys the same information.
#'
#' @inheritParams resp_headers
#' @returns `resp` (invisibly).
#' @export
#' @examples
#' resp <- request(example_url()) |>
#' req_url_path("/json") |>
#' req_perform()
#' resp |> resp_raw()
resp_raw <- function(resp) {
cli::cat_line("HTTP/1.1 ", resp$status_code, " ", resp_status_desc(resp))
cli::cat_line(cli::style_bold(names(resp$headers)), ": ", resp$headers)
cli::cat_line()
if (!is.null(resp$body)) {
cli::cat_line(resp_body_string(resp))
}
invisible(resp)
}
is_response <- function(x) {
inherits(x, "httr2_response")
}
check_response <- function(resp, arg = caller_arg(resp), call = caller_env()) {
if (!missing(resp) && is_response(resp)) {
return(invisible(NULL))
}
stop_input_type(
resp,
"an HTTP response object",
allow_null = FALSE,
arg = arg,
call = call
)
}
Add the following code to your website.
For more information on customizing the embed code, read Embedding Snippets.