2024-11-04 11:41:44 +00:00
|
|
|
use std::{thread, time::Duration};
|
|
|
|
|
2023-03-02 19:05:18 +00:00
|
|
|
use mockito::Server;
|
|
|
|
use nu_test_support::{nu, pipeline};
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn http_get_is_success() {
|
|
|
|
let mut server = Server::new();
|
|
|
|
|
|
|
|
let _mock = server.mock("GET", "/").with_body("foo").create();
|
|
|
|
|
|
|
|
let actual = nu!(pipeline(
|
|
|
|
format!(
|
|
|
|
r#"
|
|
|
|
http get {url}
|
|
|
|
"#,
|
|
|
|
url = server.url()
|
|
|
|
)
|
|
|
|
.as_str()
|
|
|
|
));
|
|
|
|
|
|
|
|
assert_eq!(actual.out, "foo")
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn http_get_failed_due_to_server_error() {
|
|
|
|
let mut server = Server::new();
|
|
|
|
|
2023-03-05 22:48:13 +00:00
|
|
|
let _mock = server.mock("GET", "/").with_status(400).create();
|
2023-03-02 19:05:18 +00:00
|
|
|
|
|
|
|
let actual = nu!(pipeline(
|
|
|
|
format!(
|
|
|
|
r#"
|
|
|
|
http get {url}
|
|
|
|
"#,
|
|
|
|
url = server.url()
|
|
|
|
)
|
|
|
|
.as_str()
|
|
|
|
));
|
|
|
|
|
|
|
|
assert!(actual.err.contains("Bad request (400)"))
|
|
|
|
}
|
2023-03-07 15:56:39 +00:00
|
|
|
|
2023-03-23 20:32:35 +00:00
|
|
|
#[test]
|
|
|
|
fn http_get_with_accept_errors() {
|
|
|
|
let mut server = Server::new();
|
|
|
|
|
|
|
|
let _mock = server
|
|
|
|
.mock("GET", "/")
|
|
|
|
.with_status(400)
|
|
|
|
.with_body("error body")
|
|
|
|
.create();
|
|
|
|
|
|
|
|
let actual = nu!(pipeline(
|
|
|
|
format!(
|
|
|
|
r#"
|
|
|
|
http get -e {url}
|
|
|
|
"#,
|
|
|
|
url = server.url()
|
|
|
|
)
|
|
|
|
.as_str()
|
|
|
|
));
|
|
|
|
|
|
|
|
assert!(actual.out.contains("error body"))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn http_get_with_accept_errors_and_full_raw_response() {
|
|
|
|
let mut server = Server::new();
|
|
|
|
|
|
|
|
let _mock = server
|
|
|
|
.mock("GET", "/")
|
|
|
|
.with_status(400)
|
|
|
|
.with_body("error body")
|
|
|
|
.create();
|
|
|
|
|
|
|
|
let actual = nu!(pipeline(
|
|
|
|
format!(
|
|
|
|
r#"
|
|
|
|
http get -e -f {url} | $"($in.status) => ($in.body)"
|
|
|
|
"#,
|
|
|
|
url = server.url()
|
|
|
|
)
|
|
|
|
.as_str()
|
|
|
|
));
|
|
|
|
|
|
|
|
assert!(actual.out.contains("400 => error body"))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn http_get_with_accept_errors_and_full_json_response() {
|
|
|
|
let mut server = Server::new();
|
|
|
|
|
|
|
|
let _mock = server
|
|
|
|
.mock("GET", "/")
|
|
|
|
.with_status(400)
|
|
|
|
.with_header("content-type", "application/json")
|
|
|
|
.with_body(
|
|
|
|
r#"
|
|
|
|
{"msg": "error body"}
|
|
|
|
"#,
|
|
|
|
)
|
|
|
|
.create();
|
|
|
|
|
|
|
|
let actual = nu!(pipeline(
|
|
|
|
format!(
|
|
|
|
r#"
|
|
|
|
http get -e -f {url} | $"($in.status) => ($in.body.msg)"
|
|
|
|
"#,
|
|
|
|
url = server.url()
|
|
|
|
)
|
|
|
|
.as_str()
|
|
|
|
));
|
|
|
|
|
|
|
|
assert!(actual.out.contains("400 => error body"))
|
|
|
|
}
|
|
|
|
|
2023-07-30 20:28:48 +00:00
|
|
|
#[test]
|
|
|
|
fn http_get_with_custom_headers_as_records() {
|
|
|
|
let mut server = Server::new();
|
|
|
|
|
|
|
|
let mock1 = server
|
|
|
|
.mock("GET", "/")
|
|
|
|
.match_header("content-type", "application/json")
|
|
|
|
.with_body(r#"{"hello": "world"}"#)
|
|
|
|
.create();
|
|
|
|
|
|
|
|
let mock2 = server
|
|
|
|
.mock("GET", "/")
|
|
|
|
.match_header("content-type", "text/plain")
|
|
|
|
.with_body("world")
|
|
|
|
.create();
|
|
|
|
|
|
|
|
let _json_response = nu!(format!(
|
|
|
|
"http get -H {{content-type: application/json}} {url}",
|
|
|
|
url = server.url()
|
|
|
|
));
|
|
|
|
|
|
|
|
let _text_response = nu!(format!(
|
|
|
|
"http get -H {{content-type: text/plain}} {url}",
|
|
|
|
url = server.url()
|
|
|
|
));
|
2023-08-17 14:19:10 +00:00
|
|
|
|
2023-07-30 20:28:48 +00:00
|
|
|
mock1.assert();
|
|
|
|
mock2.assert();
|
|
|
|
}
|
2023-08-17 14:19:10 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn http_get_full_response() {
|
|
|
|
let mut server = Server::new();
|
|
|
|
|
|
|
|
let _mock = server.mock("GET", "/").with_body("foo").create();
|
|
|
|
|
|
|
|
let actual = nu!(pipeline(
|
|
|
|
format!(
|
|
|
|
"http get --full {url} --headers [foo bar] | to json",
|
|
|
|
url = server.url()
|
|
|
|
)
|
|
|
|
.as_str()
|
|
|
|
));
|
|
|
|
|
|
|
|
let output: serde_json::Value = serde_json::from_str(&actual.out).unwrap();
|
|
|
|
|
|
|
|
assert_eq!(output["status"], 200);
|
|
|
|
assert_eq!(output["body"], "foo");
|
|
|
|
|
|
|
|
// There's only one request header, we can get it by index
|
|
|
|
assert_eq!(output["headers"]["request"][0]["name"], "foo");
|
|
|
|
assert_eq!(output["headers"]["request"][0]["value"], "bar");
|
|
|
|
|
|
|
|
// ... and multiple response headers, so have to search by name
|
|
|
|
let header = output["headers"]["response"]
|
|
|
|
.as_array()
|
|
|
|
.unwrap()
|
|
|
|
.iter()
|
|
|
|
.find(|e| e["name"] == "connection")
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(header["value"], "close");
|
|
|
|
}
|
|
|
|
|
Allow `http` commands' automatic redirect-following to be disabled (#11329)
Intends to close #8920
This PR suggests a new flag for the `http` commands, `--redirect-mode`,
which enables users to choose between different redirect handling modes.
The current behaviour of letting ureq silently follow redirects remains
the default, but two new options are introduced here, following the lead
of [JavaScript's `fetch`
API](https://developer.mozilla.org/en-US/docs/Web/API/fetch#redirect):
"manual", where any 3xx response to a request is simply returned as the
command's result, and "error", where any 3xx response causes a network
error like those caused by 4xx and 5xx responses.
This PR is a draft. Tests have not been added or run, the flag is
currently only implemented for the `http get` command, and design tweaks
are likely to be appropriate.
Most notably, it's not obvious to me whether a single flag which can
take one of three values is the nicest solution here.
We might instead consider two binary flags (like
`--no-following-redirects` and `--disallow-redirects`, although I'm bad
at naming things so I need help with that anyway), or completely drop
the "error" option if it's not deemed useful enough. (I personally think
it has some merit, especially since 4xx and 5xx responses are already
treated as errors by default; So this would allow users to treat only
immediate 2xx responses as success)
# User-facing changes
New options for the `http [method]` commands. Behaviour remains
unchanged when the command line flag introduced here is not used.
![image](https://github.com/nushell/nushell/assets/12228688/1eb89f14-7d48-4f41-8a3e-cc0f1bd0a4f8)
2023-12-28 07:26:34 +00:00
|
|
|
#[test]
|
|
|
|
fn http_get_follows_redirect() {
|
|
|
|
let mut server = Server::new();
|
|
|
|
|
|
|
|
let _mock = server.mock("GET", "/bar").with_body("bar").create();
|
|
|
|
let _mock = server
|
|
|
|
.mock("GET", "/foo")
|
|
|
|
.with_status(301)
|
|
|
|
.with_header("Location", "/bar")
|
|
|
|
.create();
|
|
|
|
|
|
|
|
let actual = nu!(pipeline(
|
|
|
|
format!("http get {url}/foo", url = server.url()).as_str()
|
|
|
|
));
|
|
|
|
|
|
|
|
assert_eq!(&actual.out, "bar");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn http_get_redirect_mode_manual() {
|
|
|
|
let mut server = Server::new();
|
|
|
|
|
|
|
|
let _mock = server
|
|
|
|
.mock("GET", "/foo")
|
|
|
|
.with_status(301)
|
|
|
|
.with_body("foo")
|
|
|
|
.with_header("Location", "/bar")
|
|
|
|
.create();
|
|
|
|
|
|
|
|
let actual = nu!(pipeline(
|
|
|
|
format!(
|
|
|
|
"http get --redirect-mode manual {url}/foo",
|
|
|
|
url = server.url()
|
|
|
|
)
|
|
|
|
.as_str()
|
|
|
|
));
|
|
|
|
|
|
|
|
assert_eq!(&actual.out, "foo");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn http_get_redirect_mode_error() {
|
|
|
|
let mut server = Server::new();
|
|
|
|
|
|
|
|
let _mock = server
|
|
|
|
.mock("GET", "/foo")
|
|
|
|
.with_status(301)
|
|
|
|
.with_body("foo")
|
|
|
|
.with_header("Location", "/bar")
|
|
|
|
.create();
|
|
|
|
|
|
|
|
let actual = nu!(pipeline(
|
|
|
|
format!(
|
|
|
|
"http get --redirect-mode error {url}/foo",
|
|
|
|
url = server.url()
|
|
|
|
)
|
|
|
|
.as_str()
|
|
|
|
));
|
|
|
|
|
|
|
|
assert!(&actual.err.contains("nu::shell::network_failure"));
|
|
|
|
assert!(&actual.err.contains(
|
|
|
|
"Redirect encountered when redirect handling mode was 'error' (301 Moved Permanently)"
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2023-03-07 15:56:39 +00:00
|
|
|
// These tests require network access; they use badssl.com which is a Google-affiliated site for testing various SSL errors.
|
|
|
|
// Revisit this if these tests prove to be flaky or unstable.
|
2024-02-28 16:28:33 +00:00
|
|
|
//
|
|
|
|
// These tests are flaky and cause CI to fail somewhat regularly. See PR #12010.
|
2023-03-07 15:56:39 +00:00
|
|
|
|
|
|
|
#[test]
|
2024-02-28 16:28:33 +00:00
|
|
|
#[ignore = "unreliable test"]
|
2023-03-07 15:56:39 +00:00
|
|
|
fn http_get_expired_cert_fails() {
|
|
|
|
let actual = nu!("http get https://expired.badssl.com/");
|
|
|
|
assert!(actual.err.contains("network_failure"));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2024-02-28 16:28:33 +00:00
|
|
|
#[ignore = "unreliable test"]
|
2023-03-07 15:56:39 +00:00
|
|
|
fn http_get_expired_cert_override() {
|
|
|
|
let actual = nu!("http get --insecure https://expired.badssl.com/");
|
|
|
|
assert!(actual.out.contains("<html>"));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2024-02-28 16:28:33 +00:00
|
|
|
#[ignore = "unreliable test"]
|
2023-03-07 15:56:39 +00:00
|
|
|
fn http_get_self_signed_fails() {
|
|
|
|
let actual = nu!("http get https://self-signed.badssl.com/");
|
|
|
|
assert!(actual.err.contains("network_failure"));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2024-02-28 16:28:33 +00:00
|
|
|
#[ignore = "unreliable test"]
|
2023-03-07 15:56:39 +00:00
|
|
|
fn http_get_self_signed_override() {
|
|
|
|
let actual = nu!("http get --insecure https://self-signed.badssl.com/");
|
|
|
|
assert!(actual.out.contains("<html>"));
|
|
|
|
}
|
2024-08-14 14:47:01 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn http_get_with_invalid_mime_type() {
|
|
|
|
let mut server = Server::new();
|
|
|
|
|
|
|
|
let _mock = server
|
|
|
|
.mock("GET", "/foo.nuon")
|
|
|
|
.with_status(200)
|
|
|
|
// `what&ever` is not a parseable MIME type
|
|
|
|
.with_header("content-type", "what&ever")
|
|
|
|
.with_body("[1 2 3]")
|
|
|
|
.create();
|
|
|
|
|
|
|
|
// but `from nuon` is a known command in nu, so we take `foo.{ext}` and pass it to `from {ext}`
|
|
|
|
let actual = nu!(pipeline(
|
|
|
|
format!(
|
|
|
|
r#"http get {url}/foo.nuon | to json --raw"#,
|
|
|
|
url = server.url()
|
|
|
|
)
|
|
|
|
.as_str()
|
|
|
|
));
|
|
|
|
|
|
|
|
assert_eq!(actual.out, "[1,2,3]");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn http_get_with_unknown_mime_type() {
|
|
|
|
let mut server = Server::new();
|
|
|
|
let _mock = server
|
|
|
|
.mock("GET", "/foo")
|
|
|
|
.with_status(200)
|
|
|
|
// `application/nuon` is not an IANA-registered MIME type
|
|
|
|
.with_header("content-type", "application/nuon")
|
|
|
|
.with_body("[1 2 3]")
|
|
|
|
.create();
|
|
|
|
|
|
|
|
// but `from nuon` is a known command in nu, so we take `{garbage}/{whatever}` and pass it to `from {whatever}`
|
|
|
|
let actual = nu!(pipeline(
|
|
|
|
format!(r#"http get {url}/foo | to json --raw"#, url = server.url()).as_str()
|
|
|
|
));
|
|
|
|
|
|
|
|
assert_eq!(actual.out, "[1,2,3]");
|
|
|
|
}
|
2024-11-04 11:41:44 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn http_get_timeout() {
|
|
|
|
let mut server = Server::new();
|
|
|
|
let _mock = server
|
|
|
|
.mock("GET", "/")
|
|
|
|
.with_chunked_body(|w| {
|
|
|
|
thread::sleep(Duration::from_secs(1));
|
|
|
|
w.write_all(b"Delayed response!")
|
|
|
|
})
|
|
|
|
.create();
|
|
|
|
|
|
|
|
let actual = nu!(pipeline(
|
|
|
|
format!("http get --max-time 500ms {url}", url = server.url()).as_str()
|
|
|
|
));
|
|
|
|
|
|
|
|
assert!(&actual.err.contains("nu::shell::io_error"));
|
|
|
|
}
|