use super::preview_request;
use crate::commands::dev::utils::{get_path_as_str, rewrite_redirect};
use crate::commands::dev::{self, tls, Protocol, ServerConfig};
use crate::terminal::emoji;
use crate::terminal::message::{Message, StdOut};
use std::sync::{Arc, Mutex};
use anyhow::Result;
use chrono::prelude::*;
use futures_util::{stream::StreamExt, FutureExt};
use hyper::service::{make_service_fn, service_fn};
use hyper::upgrade::OnUpgrade;
use hyper::Server;
use tokio::net::TcpListener;
use tokio::sync::oneshot::{Receiver, Sender};
pub async fn https(
server_config: ServerConfig,
preview_token: Arc<Mutex<String>>,
host: String,
shutdown_channel: (Receiver<()>, Sender<()>),
) -> Result<()> {
tls::generate_cert()?;
let client = dev::client();
let listening_address = server_config.listening_address;
let service = make_service_fn(move |_| {
let client = client.to_owned();
let preview_token = preview_token.to_owned();
let host = host.to_owned();
let server_config = server_config.to_owned();
async move {
Ok::<_, anyhow::Error>(service_fn(move |req| {
let is_websocket = req
.headers()
.get("upgrade")
.map_or(false, |h| h.as_bytes() == b"websocket");
let client = client.to_owned();
let preview_token = preview_token.lock().unwrap().to_owned();
let host = host.to_owned();
let version = req.version();
let (parts, body) = req.into_parts();
let local_host = format!(
"{}:{}",
server_config.listening_address.ip().to_string(),
server_config.listening_address.port().to_string()
);
let req_method = parts.method.to_string();
let now: DateTime<Local> = Local::now();
let path = get_path_as_str(&parts.uri);
async move {
let mut req = preview_request(
parts,
body,
preview_token.to_owned(),
host.clone(),
Protocol::Http,
);
let client_on_upgrade = req.extensions_mut().remove::<OnUpgrade>();
let mut resp = client.request(req).await?;
super::maybe_proxy_websocket(is_websocket, client_on_upgrade, &mut resp);
rewrite_redirect(&mut resp, &host, &local_host, true);
println!(
"[{}] {} {}{} {:?} {}",
now.format("%Y-%m-%d %H:%M:%S"),
req_method,
host,
path,
version,
resp.status()
);
Ok::<_, anyhow::Error>(resp)
}
}))
}
});
let tcp = TcpListener::bind(&listening_address).await?;
let tls_acceptor = &tls::get_tls_acceptor()?;
let incoming_tls_stream = async {
let tcp_stream = match tcp.accept().await {
Ok((tcp_stream, _addr)) => Ok(tcp_stream),
Err(e) => {
eprintln!("Failed to accept client {}", e);
Err(e)
}
};
match tcp_stream {
Ok(stream) => match tls_acceptor.accept(stream).await {
Ok(tls_stream) => Ok(tls_stream),
Err(e) => {
eprintln!("Client connection error {}", e);
StdOut::info("Make sure to use https and `--insecure` with curl");
Err(e)
}
},
Err(e) => Err(e),
}
}
.into_stream()
.boxed();
let (rx, tx) = shutdown_channel;
let server = Server::builder(tls::HyperAcceptor {
acceptor: incoming_tls_stream,
})
.serve(service)
.with_graceful_shutdown(async {
rx.await.expect("Could not receive shutdown initiation");
});
println!("{} Listening on https://{}", emoji::EAR, listening_address);
StdOut::info("Generated certificate is not verified, browsers will give a warning and curl will require `--insecure`");
if let Err(e) = server.await {
eprintln!("{}", e);
}
if let Err(e) = tx.send(()) {
log::error!("Could not acknowledge dev https listener shutdown: {:?}", e);
}
Ok(())
}