summaryrefslogtreecommitdiff
path: root/vendor/hyper-rustls/examples/client.rs
diff options
context:
space:
mode:
Diffstat (limited to 'vendor/hyper-rustls/examples/client.rs')
-rw-r--r--vendor/hyper-rustls/examples/client.rs105
1 files changed, 105 insertions, 0 deletions
diff --git a/vendor/hyper-rustls/examples/client.rs b/vendor/hyper-rustls/examples/client.rs
new file mode 100644
index 00000000..c45bc2a7
--- /dev/null
+++ b/vendor/hyper-rustls/examples/client.rs
@@ -0,0 +1,105 @@
+//! Simple HTTPS GET client based on hyper-rustls
+//!
+//! First parameter is the mandatory URL to GET.
+//! Second parameter is an optional path to CA store.
+use http::Uri;
+use http_body_util::{BodyExt, Empty};
+use hyper::body::Bytes;
+use hyper_rustls::ConfigBuilderExt;
+use hyper_util::{client::legacy::Client, rt::TokioExecutor};
+use rustls::RootCertStore;
+
+use std::str::FromStr;
+use std::{env, fs, io};
+
+fn main() {
+ // Send GET request and inspect result, with proper error handling.
+ if let Err(e) = run_client() {
+ eprintln!("FAILED: {}", e);
+ std::process::exit(1);
+ }
+}
+
+fn error(err: String) -> io::Error {
+ io::Error::new(io::ErrorKind::Other, err)
+}
+
+#[tokio::main]
+async fn run_client() -> io::Result<()> {
+ // Set a process wide default crypto provider.
+ #[cfg(feature = "ring")]
+ let _ = rustls::crypto::ring::default_provider().install_default();
+ #[cfg(feature = "aws-lc-rs")]
+ let _ = rustls::crypto::aws_lc_rs::default_provider().install_default();
+
+ // First parameter is target URL (mandatory).
+ let url = match env::args().nth(1) {
+ Some(ref url) => Uri::from_str(url).map_err(|e| error(format!("{}", e)))?,
+ None => {
+ println!("Usage: client <url> <ca_store>");
+ return Ok(());
+ }
+ };
+
+ // Second parameter is custom Root-CA store (optional, defaults to native cert store).
+ let mut ca = match env::args().nth(2) {
+ Some(ref path) => {
+ let f = fs::File::open(path)
+ .map_err(|e| error(format!("failed to open {}: {}", path, e)))?;
+ let rd = io::BufReader::new(f);
+ Some(rd)
+ }
+ None => None,
+ };
+
+ // Prepare the TLS client config
+ let tls = match ca {
+ Some(ref mut rd) => {
+ // Read trust roots
+ let certs = rustls_pemfile::certs(rd).collect::<Result<Vec<_>, _>>()?;
+ let mut roots = RootCertStore::empty();
+ roots.add_parsable_certificates(certs);
+ // TLS client config using the custom CA store for lookups
+ rustls::ClientConfig::builder()
+ .with_root_certificates(roots)
+ .with_no_client_auth()
+ }
+ // Default TLS client config with native roots
+ None => rustls::ClientConfig::builder()
+ .with_native_roots()?
+ .with_no_client_auth(),
+ };
+ // Prepare the HTTPS connector
+ let https = hyper_rustls::HttpsConnectorBuilder::new()
+ .with_tls_config(tls)
+ .https_or_http()
+ .enable_http1()
+ .build();
+
+ // Build the hyper client from the HTTPS connector.
+ let client: Client<_, Empty<Bytes>> = Client::builder(TokioExecutor::new()).build(https);
+
+ // Prepare a chain of futures which sends a GET request, inspects
+ // the returned headers, collects the whole body and prints it to
+ // stdout.
+ let fut = async move {
+ let res = client
+ .get(url)
+ .await
+ .map_err(|e| error(format!("Could not get: {:?}", e)))?;
+ println!("Status:\n{}", res.status());
+ println!("Headers:\n{:#?}", res.headers());
+
+ let body = res
+ .into_body()
+ .collect()
+ .await
+ .map_err(|e| error(format!("Could not get body: {:?}", e)))?
+ .to_bytes();
+ println!("Body:\n{}", String::from_utf8_lossy(&body));
+
+ Ok(())
+ };
+
+ fut.await
+}