From 6a612ca388ec3b383db232b68c58ef5a5b54a3e2 Mon Sep 17 00:00:00 2001 From: Grant Handy Date: Mon, 25 Sep 2023 01:46:36 -0600 Subject: [PATCH] Simple console error message dialog in `zola serve` webpages (#2312) * first attempt at injecting HTML into outgoing webpages * finish error handling * fix test_site debug line --- src/cmd/serve.rs | 92 +++++++++++++++++++++++++++++++++++++++++++++--- 1 file changed, 88 insertions(+), 4 deletions(-) diff --git a/src/cmd/serve.rs b/src/cmd/serve.rs index 52b1d585..42b79be8 100644 --- a/src/cmd/serve.rs +++ b/src/cmd/serve.rs @@ -21,16 +21,20 @@ // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. +use std::cell::Cell; use std::fs::read_dir; +use std::future::IntoFuture; use std::net::{SocketAddrV4, TcpListener}; use std::path::{Path, PathBuf, MAIN_SEPARATOR}; use std::sync::mpsc::channel; +use std::sync::Mutex; use std::thread; use std::time::{Duration, Instant}; -use hyper::header; +use hyper::http::HeaderValue; use hyper::server::Server; use hyper::service::{make_service_fn, service_fn}; +use hyper::{body, header}; use hyper::{Body, Method, Request, Response, StatusCode}; use mime_guess::from_path as mimetype_from_path; use time::macros::format_description; @@ -75,6 +79,19 @@ static NOT_FOUND_TEXT: &[u8] = b"Not Found"; // This is dist/livereload.min.js from the LiveReload.js v3.2.4 release const LIVE_RELOAD: &str = include_str!("livereload.js"); +static SERVE_ERROR: Mutex>> = + Mutex::new(Cell::new(None)); + +fn clear_serve_error() { + let _ = SERVE_ERROR.lock().map(|error| error.swap(&Cell::new(None))); +} + +fn set_serve_error(msg: &'static str, e: errors::Error) { + if let Ok(serve_error) = SERVE_ERROR.lock() { + serve_error.swap(&Cell::new(Some((msg, e)))); + } +} + async fn handle_request(req: Request, mut root: PathBuf) -> Result> { let original_root = root.clone(); let mut path = RelativePathBuf::new(); @@ -148,6 +165,61 @@ async fn handle_request(req: Request, mut root: PathBuf) -> Result>>, +) -> Result> { + let req = req.await; + + // return req as-is if the request is Err(), not HTML, or if there are no error messages. + if req + .as_ref() + .map(|req| { + req.headers() + .get(header::CONTENT_TYPE) + .map(|val| val != &HeaderValue::from_static("text/html")) + .unwrap_or(true) + }) + .unwrap_or(true) + || SERVE_ERROR.lock().unwrap().get_mut().is_none() + { + return req; + } + + let mut req = req.unwrap(); + let mut bytes = body::to_bytes(req.body_mut()).await.unwrap().to_vec(); + + if let Some((msg, error)) = SERVE_ERROR.lock().unwrap().get_mut() { + // Generate an error message similar to the CLI version in messages::unravel_errors. + let mut error_str = String::new(); + + if !msg.is_empty() { + error_str.push_str(&format!("Error: {msg}\n")); + } + + error_str.push_str(&format!("Error: {error}\n")); + + let mut cause = error.source(); + while let Some(e) = cause { + error_str.push_str(&format!("Reason: {}\n", e)); + cause = e.source(); + } + + // Push the error message (wrapped in an HTML dialog box) to the end of the HTML body. + // + // The message will be outside of and but web browsers are flexible enough + // that they will move it to the end of at page load. + let html_error = format!( + r#"

Zola Build Error:

{error_str}
"# + ); + bytes.extend(html_error.as_bytes()); + + *req.body_mut() = Body::from(bytes); + } + + Ok(req) +} + fn livereload_js() -> Response { Response::builder() .header(header::CONTENT_TYPE, "text/javascript") @@ -213,6 +285,7 @@ fn not_found() -> Response { fn rebuild_done_handling(broadcaster: &Sender, res: Result<()>, reload_path: &str) { match res { Ok(_) => { + clear_serve_error(); broadcaster .send(format!( r#" @@ -228,7 +301,12 @@ fn rebuild_done_handling(broadcaster: &Sender, res: Result<()>, reload_path: &st )) .unwrap(); } - Err(e) => messages::unravel_errors("Failed to build the site", &e), + Err(e) => { + let msg = "Failed to build the site"; + + messages::unravel_errors(msg, &e); + set_serve_error(msg, e); + } } } @@ -384,7 +462,7 @@ pub fn serve( async { Ok::<_, hyper::Error>(service_fn(move |req| { - handle_request(req, static_root.clone()) + response_error_injector(handle_request(req, static_root.clone())) })) } }); @@ -514,11 +592,17 @@ pub fn serve( ws_port, ) { Ok((s, _)) => { + clear_serve_error(); rebuild_done_handling(&broadcaster, Ok(()), "/x.js"); + Some(s) } Err(e) => { - messages::unravel_errors("Failed to build the site", &e); + let msg = "Failed to build the site"; + + messages::unravel_errors(msg, &e); + set_serve_error(msg, e); + None } };