use crate::template_args; use crate::State; use actix_web::get; use actix_web::http::header; use actix_web::web::{Data, Query}; use actix_web::{HttpRequest, HttpResponse, Responder}; use handlebars::Handlebars; use itertools::Itertools; use log::{debug, error}; use percent_encoding::{utf8_percent_encode, AsciiSet, CONTROLS}; use serde::{de::Visitor, Deserialize, Deserializer, Serialize, Serializer}; use std::collections::HashMap; use std::fmt; use std::path::PathBuf; use std::process::Command; use std::sync::{Arc, RwLock}; /// https://url.spec.whatwg.org/#fragment-percent-encode-set const FRAGMENT_ENCODE_SET: &AsciiSet = &CONTROLS .add(b' ') .add(b'"') .add(b'<') .add(b'>') .add(b'`') .add(b'+'); type StateData = Data>>; #[derive(Debug, PartialEq, Clone)] pub enum Route { External(String), Path(String), } /// Serialization of the Route enum needs to be transparent, but since the /// `#[serde(transparent)]` macro isn't available on enums, so we need to /// implement it manually. impl Serialize for Route { fn serialize(&self, serializer: S) -> Result where S: Serializer, { match self { Self::External(s) => serializer.serialize_str(s), Self::Path(s) => serializer.serialize_str(s), } } } /// Deserialization of the route string into the enum requires us to figure out /// whether or not the string is valid to run as an executable or not. To /// determine this, we simply check if it exists on disk or assume that it's a /// web path. This incurs a disk check operation, but since users shouldn't be /// updating the config that frequently, it should be fine. impl<'de> Deserialize<'de> for Route { fn deserialize(deserializer: D) -> Result where D: Deserializer<'de>, { struct RouteVisitor; impl<'de> Visitor<'de> for RouteVisitor { type Value = Route; fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result { formatter.write_str("string") } fn visit_str(self, value: &str) -> Result where E: serde::de::Error, { // Return early if it's a path, don't go through URL parsing if std::path::Path::new(value).exists() { debug!("Parsed {} as a valid local path.", value); Ok(Route::Path(value.into())) } else { debug!("{} does not exist on disk, assuming web path.", value); Ok(Route::External(value.into())) } } } deserializer.deserialize_str(RouteVisitor) } } impl std::fmt::Display for Route { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { match self { Self::External(s) => write!(f, "raw ({})", s), Self::Path(s) => write!(f, "file ({})", s), } } } #[get("/ls")] pub async fn list( data: Data>>, req: HttpRequest, ) -> impl Responder { let data = data.read().unwrap(); HttpResponse::Ok().body( req .app_data::() .unwrap() .render("list", &data.groups) .unwrap(), ) } #[derive(Deserialize)] pub struct SearchQuery { to: String, } #[get("/hop")] pub async fn hop( data: StateData, req: HttpRequest, query: Query, ) -> impl Responder { let data = data.read().unwrap(); match resolve_hop(&query.to, &data.routes, &data.default_route) { (Some(path), args) => { let resolved_template = match path { Route::Path(path) => resolve_path(PathBuf::from(path), &args), Route::External(path) => Ok(path.to_owned().into_bytes()), }; match resolved_template { Ok(path) => HttpResponse::Found() .header( header::LOCATION, req .app_data::() .unwrap() .render_template( std::str::from_utf8(&path).unwrap(), &template_args::query( utf8_percent_encode(&args, FRAGMENT_ENCODE_SET).to_string(), ), ) .unwrap(), ) .finish(), Err(e) => { error!("Failed to redirect user for {}: {}", path, e); HttpResponse::InternalServerError().body("Something went wrong :(\n") } } } (None, _) => HttpResponse::NotFound().body("not found"), } } /// Attempts to resolve the provided string into its route and its arguments. /// If a default route was provided, then this will consider that route before /// failing to resolve a route. /// /// The first element in the tuple describes the route, while the second element /// returns the remaining arguments. If none remain, an empty string is given. fn resolve_hop<'a>( query: &str, routes: &'a HashMap, default_route: &Option, ) -> (Option<&'a Route>, String) { let mut split_args = query.split_ascii_whitespace().peekable(); let command = match split_args.peek() { Some(command) => command, None => { debug!("Found empty query, returning no route."); return (None, String::new()); } }; match (routes.get(*command), default_route) { // Found a route (Some(resolved), _) => ( Some(resolved), match split_args.next() { // Discard the first result, we found the route using the first arg Some(_) => { let args = split_args.join(" "); debug!("Resolved {} with args {}", resolved, args); args } None => { debug!("Resolved {} with no args", resolved); String::new() } }, ), // Unable to find route, but had a default route (None, Some(route)) => { let args = split_args.join(" "); debug!("Using default route {} with args {}", route, args); match routes.get(route) { Some(v) => (Some(v), args), None => (None, String::new()), } } // No default route and no match (None, None) => { debug!("Failed to resolve route!"); (None, String::new()) } } } #[get("/")] pub async fn index(data: StateData, req: HttpRequest) -> impl Responder { let data = data.read().unwrap(); HttpResponse::Ok().body( req .app_data::() .unwrap() .render( "index", &template_args::hostname(data.public_address.clone()), ) .unwrap(), ) } /// Runs the executable with the user's input as a single argument. Returns Ok /// so long as the executable was successfully executed. Returns an Error if the /// file doesn't exist or bunbun did not have permission to read and execute the /// file. fn resolve_path( path: PathBuf, args: &str, ) -> Result, crate::BunBunError> { let output = Command::new(path.canonicalize()?).arg(args).output()?; if output.status.success() { Ok(output.stdout) } else { error!( "Program exit code for {} was not 0! Dumping standard error!", path.display(), ); let error = String::from_utf8_lossy(&output.stderr); Err(crate::BunBunError::CustomProgramError(error.to_string())) } } #[get("/bunbunsearch.xml")] pub async fn opensearch(data: StateData, req: HttpRequest) -> impl Responder { let data = data.read().unwrap(); HttpResponse::Ok() .header( header::CONTENT_TYPE, "application/opensearchdescription+xml", ) .body( req .app_data::() .unwrap() .render( "opensearch", &template_args::hostname(data.public_address.clone()), ) .unwrap(), ) } #[cfg(test)] mod route { use super::*; use serde_yaml::{from_str, to_string}; use tempfile::NamedTempFile; #[test] fn deserialize_relative_path() { let tmpfile = NamedTempFile::new_in(".").unwrap(); let path = format!("{}", tmpfile.path().display()); let path = path.get(path.rfind(".").unwrap()..).unwrap(); let path = std::path::Path::new(path); assert!(path.is_relative()); let path = path.to_str().unwrap(); assert_eq!(from_str::(path).unwrap(), Route::Path(path.into())); } #[test] fn deserialize_absolute_path() { let tmpfile = NamedTempFile::new().unwrap(); let path = format!("{}", tmpfile.path().display()); assert!(tmpfile.path().is_absolute()); assert_eq!(from_str::(&path).unwrap(), Route::Path(path)); } #[test] fn deserialize_http_path() { assert_eq!( from_str::("http://google.com").unwrap(), Route::External("http://google.com".into()) ); } #[test] fn deserialize_https_path() { assert_eq!( from_str::("https://google.com").unwrap(), Route::External("https://google.com".into()) ); } #[test] fn serialize() { assert_eq!( &to_string(&Route::External("hello world".into())).unwrap(), "---\nhello world" ); assert_eq!( &to_string(&Route::Path("hello world".into())).unwrap(), "---\nhello world" ); } } #[cfg(test)] mod resolve_hop { use super::*; fn generate_route_result<'a>( keyword: &'a Route, args: &str, ) -> (Option<&'a Route>, String) { (Some(keyword), String::from(args)) } #[test] fn empty_routes_no_default_yields_failed_hop() { assert_eq!( resolve_hop("hello world", &HashMap::new(), &None), (None, String::new()) ); } #[test] fn empty_routes_some_default_yields_failed_hop() { assert_eq!( resolve_hop( "hello world", &HashMap::new(), &Some(String::from("google")) ), (None, String::new()) ); } #[test] fn only_default_routes_some_default_yields_default_hop() { let mut map: HashMap = HashMap::new(); map.insert( "google".into(), Route::External("https://example.com".into()), ); assert_eq!( resolve_hop("hello world", &map, &Some(String::from("google"))), generate_route_result( &Route::External("https://example.com".into()), "hello world" ), ); } #[test] fn non_default_routes_some_default_yields_non_default_hop() { let mut map: HashMap = HashMap::new(); map.insert( "google".into(), Route::External("https://example.com".into()), ); assert_eq!( resolve_hop("google hello world", &map, &Some(String::from("a"))), generate_route_result( &Route::External("https://example.com".into()), "hello world" ), ); } #[test] fn non_default_routes_no_default_yields_non_default_hop() { let mut map: HashMap = HashMap::new(); map.insert( "google".into(), Route::External("https://example.com".into()), ); assert_eq!( resolve_hop("google hello world", &map, &None), generate_route_result( &Route::External("https://example.com".into()), "hello world" ), ); } }