handle uri parameters correctly

This commit is contained in:
Dietmar Maurer 2018-11-16 09:15:33 +01:00
parent 484eb9ac19
commit e7ea17deb0
3 changed files with 29 additions and 20 deletions

View File

@ -21,10 +21,9 @@ impl ApiConfig {
} }
} }
pub fn find_method(&self, components: &[&str], method: Method) -> Option<&'static ApiMethod> { pub fn find_method(&self, components: &[&str], method: Method, uri_param: &mut HashMap<String, String>) -> Option<&'static ApiMethod> {
if let Some(info) = self.router.find_route(components) { if let Some(info) = self.router.find_route(components, uri_param) {
println!("FOUND INFO");
let opt_api_method = match method { let opt_api_method = match method {
Method::GET => &info.get, Method::GET => &info.get,
Method::PUT => &info.put, Method::PUT => &info.put,

View File

@ -67,7 +67,7 @@ impl Router {
self self
} }
pub fn find_route(&self, components: &[&str]) -> Option<&Router> { pub fn find_route(&self, components: &[&str], uri_param: &mut HashMap<String, String>) -> Option<&Router> {
if components.len() == 0 { return Some(self); }; if components.len() == 0 { return Some(self); };
@ -78,12 +78,13 @@ impl Router {
SubRoute::Hash(ref dirmap) => { SubRoute::Hash(ref dirmap) => {
if let Some(ref router) = dirmap.get(dir) { if let Some(ref router) = dirmap.get(dir) {
println!("FOUND SUBDIR {}", dir); println!("FOUND SUBDIR {}", dir);
return router.find_route(rest); return router.find_route(rest, uri_param);
} }
} }
SubRoute::MatchAll { ref router, ref param_name } => { SubRoute::MatchAll { ref router, ref param_name } => {
println!("URI PARAM {} = {}", param_name, dir); // fixme: store somewhere println!("URI PARAM {} = {}", param_name, dir); // fixme: store somewhere
return router.find_route(rest); uri_param.insert(param_name.clone(), dir.into());
return router.find_route(rest, uri_param);
}, },
} }

View File

@ -5,9 +5,11 @@ use crate::api::config::*;
use std::fmt; use std::fmt;
use std::path::{PathBuf}; use std::path::{PathBuf};
use std::sync::Arc; use std::sync::Arc;
use std::collections::HashMap;
use failure::*; use failure::*;
use serde_json::{json, Value}; use serde_json::{Value};
use url::form_urlencoded;
use futures::future::{self, Either}; use futures::future::{self, Either};
//use tokio::prelude::*; //use tokio::prelude::*;
@ -109,6 +111,7 @@ fn get_request_parameters_async(
info: &'static ApiMethod, info: &'static ApiMethod,
parts: Parts, parts: Parts,
req_body: Body, req_body: Body,
uri_param: HashMap<String, String>,
) -> Box<Future<Item = Value, Error = failure::Error> + Send> ) -> Box<Future<Item = Value, Error = failure::Error> + Send>
{ {
let resp = req_body let resp = req_body
@ -126,23 +129,27 @@ fn get_request_parameters_async(
println!("GOT BODY {:?}", bytes); println!("GOT BODY {:?}", bytes);
let mut test_required = true; let mut param_list: Vec<(String, String)> = vec![];
let mut params = json!({});
if bytes.len() > 0 { if bytes.len() > 0 {
params = parse_query_string(&bytes, &info.parameters, true)?; for (k, v) in form_urlencoded::parse(bytes.as_bytes()).into_owned() {
test_required = false; param_list.push((k, v));
}
} }
if let Some(query_str) = parts.uri.query() { if let Some(query_str) = parts.uri.query() {
let query_params = parse_query_string(query_str, &info.parameters, test_required)?; for (k, v) in form_urlencoded::parse(query_str.as_bytes()).into_owned() {
param_list.push((k, v));
for (k, v) in query_params.as_object().unwrap() {
params[k] = v.clone(); // fixme: why clone()??
} }
} }
for (k, v) in uri_param {
param_list.push((k.clone(), v.clone()));
}
let params = parse_parameter_strings(&param_list, &info.parameters, true)?;
println!("GOT PARAMS {}", params); println!("GOT PARAMS {}", params);
Ok(params) Ok(params)
}); });
@ -154,9 +161,10 @@ fn handle_sync_api_request(
info: &'static ApiMethod, info: &'static ApiMethod,
parts: Parts, parts: Parts,
req_body: Body, req_body: Body,
uri_param: HashMap<String, String>,
) -> BoxFut ) -> BoxFut
{ {
let params = get_request_parameters_async(info, parts, req_body); let params = get_request_parameters_async(info, parts, req_body, uri_param);
let resp = params let resp = params
.and_then(move |params| { .and_then(move |params| {
@ -278,9 +286,11 @@ pub fn handle_request(api: Arc<ApiConfig>, req: Request<Body>) -> BoxFut {
return Box::new(future::err(http_err!(BAD_REQUEST, format!("Unsupported output format '{}'.", format)))) return Box::new(future::err(http_err!(BAD_REQUEST, format!("Unsupported output format '{}'.", format))))
} }
if let Some(api_method) = api.find_method(&components[2..], method) { let mut uri_param = HashMap::new();
if let Some(api_method) = api.find_method(&components[2..], method, &mut uri_param) {
// fixme: handle auth // fixme: handle auth
return handle_sync_api_request(api_method, parts, body); return handle_sync_api_request(api_method, parts, body, uri_param);
} }
} }
} else { } else {
@ -292,4 +302,3 @@ pub fn handle_request(api: Arc<ApiConfig>, req: Request<Body>) -> BoxFut {
Box::new(future::err(http_err!(NOT_FOUND, "Path not found.".to_string()))) Box::new(future::err(http_err!(NOT_FOUND, "Path not found.".to_string())))
} }