use crate::barbfile::{BarbFile, BarbFilter}; use crate::output::BarbOutput; use jq_rs; use std::collections::HashMap; use ureq; use ureq::Error as UreqError; use std::fs; use std::str::FromStr; pub struct Context { vars: HashMap, } impl Context { pub fn new(vars: I) -> Context where I: Iterator, { let mut toto = HashMap::new(); toto.extend(vars); Context { vars: toto } } #[cfg(test)] pub fn empty() -> Context { Context { vars: HashMap::new(), } } // Preserved for future reference // pub fn get_var(&self, name: String) -> Option { // self.vars // .get(&name) // .map(|val| val.clone()) // .or_else(|| env::var(name).ok()) // } fn key_str(&self, key: &String) -> String { format!("{{{}}}", key) } pub fn substitute(&self, string: &String) -> String { let mut buffer = string.clone(); for (key, val) in self.vars.iter() { buffer = buffer.replace(self.key_str(key).as_str(), val); } buffer } } fn apply_filter(filter: &BarbFilter, body: &String) -> Result { jq_rs::run(filter.filter().as_str(), body.as_str()).map_err(|x| x.to_string()) } fn apply_filters(bfile: &BarbFile, body: String, arg_filter: &Option) -> Result { if let Some(filter) = arg_filter { return Ok(apply_filter(&BarbFilter::new(None, filter.to_string()), &body)?); } else if bfile.filters().len() > 0 { return Ok(bfile .filters() .iter() .map(|filter| apply_filter(filter, &body)) .last() .unwrap().unwrap()); } Ok(String::from(body)) } pub struct Executor { context: Context, } impl Executor { pub fn new(context: Context) -> Executor { Executor { context } } fn make_req(&self, bfile: &BarbFile, output: &BarbOutput) -> ureq::Request { output.req( bfile.method_as_string(), self.context.substitute(&bfile.url()), ); let mut req = ureq::request( bfile.method_as_string().as_str(), self.context.substitute(&bfile.url()).as_str(), ); for header in bfile.headers() { let hdr_val = self.context.substitute(header.value()); req = req.set(header.name(), hdr_val.as_str()); output.req_hdr(header.name().to_string(), hdr_val); } output.end_req(); req } fn run(&self, bfile: &BarbFile, req: ureq::Request) -> Result { let resp = match bfile.method().takes_body() { true => match bfile.body() { Some(body) => req.send_string(body.as_str()), None => req.call(), }, false => req.call(), }; match resp { Ok(resp) => Ok(resp), Err(UreqError::Status(_, resp)) => Ok(resp), Err(UreqError::Transport(transp)) => Err(String::from(transp.to_string())), } } pub fn execute( &mut self, file_name: &String, output: &BarbOutput, filter: &Option ) -> Result<(), String> { let bfile = BarbFile::from_str( fs::read_to_string(file_name.as_str()) .map_err(|_| format!("Failed to read file '{}'", file_name))? .as_str(), ) .map_err(|_| format!("Failed to parse file '{}'", file_name))?; let response = self.run(&bfile, self.make_req(&bfile, output))?; //let response = executor.execute(&bfile, &output)?; output.status(response.status(), response.status_text()); for header_name in response.headers_names() { output.resp_hdr( header_name.to_string(), response.header(header_name.as_str()).unwrap(), ); } output.end_resp_hdr(); output.body(apply_filters( &bfile, response.into_string().unwrap(), filter )?); Ok(()) } } // TODO: tests #[cfg(test)] mod tests { use super::*; use std::str::FromStr; #[test] fn test_context_key_str() { let ctx = Context::empty(); assert_eq!(ctx.key_str(&String::from("foo")), String::from("{foo}")); } #[test] fn test_context_substitute() { let vars: Vec<(String, String)> = vec![ (String::from("foo"), String::from("bar")), (String::from("bar"), String::from("baz")), ]; let ctx = Context::new(vars.into_iter()); assert_eq!( ctx.substitute(&String::from("blah blah {foo} blah")), String::from("blah blah bar blah") ); assert_eq!( ctx.substitute(&String::from("blah {foo} {bar} blah")), String::from("blah bar baz blah") ); } #[test] fn test_make_req_simple_get() { let executor = Executor::new(Context::empty()); let bfile = BarbFile::from_str("#GET^http://foo.bar\n\n").unwrap(); let output = BarbOutput::new(false, false, false, false, false, false); let req = executor.make_req(&bfile, &output); assert_eq!(req.url(), "http://foo.bar"); assert_eq!(req.method(), "GET"); } #[test] fn test_make_req_simple_headers() { let executor = Executor::new(Context::empty()); let bfile = BarbFile::from_str("#GET^http://foo.bar\n#Foo: Bar\n#Bar: Baz\n\n").unwrap(); let output = BarbOutput::new(false, false, false, false, false, false); let req = executor.make_req(&bfile, &output); assert_eq!( req.header_names(), vec![String::from("foo"), String::from("bar")] ); assert_eq!(req.header("foo"), Some("Bar")); assert_eq!(req.header("bar"), Some("Baz")); } }