400 lines
11 KiB
Rust
400 lines
11 KiB
Rust
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<Arc<RwLock<State>>>;
|
|
|
|
#[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<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
|
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<D>(deserializer: D) -> Result<Route, D::Error>
|
|
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<E>(self, value: &str) -> Result<Self::Value, E>
|
|
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<Arc<RwLock<State>>>,
|
|
req: HttpRequest,
|
|
) -> impl Responder {
|
|
let data = data.read().unwrap();
|
|
HttpResponse::Ok().body(
|
|
req
|
|
.app_data::<Handlebars>()
|
|
.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<SearchQuery>,
|
|
) -> 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::<Handlebars>()
|
|
.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<String, Route>,
|
|
default_route: &Option<String>,
|
|
) -> (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::<Handlebars>()
|
|
.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<Vec<u8>, 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::<Handlebars>()
|
|
.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::<Route>(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::<Route>(&path).unwrap(), Route::Path(path));
|
|
}
|
|
|
|
#[test]
|
|
fn deserialize_http_path() {
|
|
assert_eq!(
|
|
from_str::<Route>("http://google.com").unwrap(),
|
|
Route::External("http://google.com".into())
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn deserialize_https_path() {
|
|
assert_eq!(
|
|
from_str::<Route>("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<String, Route> = 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<String, Route> = 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<String, Route> = 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"
|
|
),
|
|
);
|
|
}
|
|
}
|