use std::collections::HashMap;
use crate::routekey::{Params, Segments};
#[derive(Clone, Copy, Hash, PartialEq, Eq)]
pub(crate) struct RouteId(usize);
#[derive(Clone)]
pub struct Router<V>
where
V: Clone,
{
routes: Vec<(Segments, V)>,
lookup_cache: HashMap<Segments, RouteId>,
}
impl<V> Default for Router<V>
where
V: Clone,
{
fn default() -> Self {
Self { routes: Vec::default(), lookup_cache: HashMap::default() }
}
}
pub struct Match<V> {
pub value: V,
pub params: HashMap<String, String>,
}
impl<V> Router<V>
where
V: Clone,
{
pub fn at(&mut self, route: &str, handler: V) {
let segments: Segments = route.to_string().into();
let index = self.routes.len(); let route_id = RouteId(index);
self.routes.push((segments.clone(), handler));
if segments.num_params() == 0 {
self.lookup_cache.insert(segments, route_id);
}
}
pub fn find(&mut self, route: &str) -> Option<Match<&V>> {
let segments = Segments::from(route.to_string());
if let Some(RouteId(route_index)) = self.lookup_cache.get(&segments) {
let (_, value) = &self.routes[*route_index];
return Some(Match { value, params: HashMap::default() });
};
for (index, (key, value)) in self.routes.iter().enumerate() {
if let Some(Params(params)) = key.find(&segments.0) {
if params.is_empty() {
self.lookup_cache.insert(segments, RouteId(index));
}
return Some(Match { value, params });
};
}
None
}
pub fn lookup(&self, route: &str) -> Option<Match<&V>> {
let segments = Segments::from(route.to_string());
if let Some(RouteId(route_index)) = self.lookup_cache.get(&segments) {
let (_, value) = &self.routes[*route_index];
return Some(Match { value, params: HashMap::default() });
};
for (key, value) in &self.routes {
if let Some(Params(params)) = key.find(&segments.0) {
return Some(Match { value, params });
};
}
None
}
pub fn at_mut(&mut self, route: &str) -> Option<Match<&mut V>> {
let segments = Segments::from(route.to_string());
if let Some(RouteId(route_index)) = self.lookup_cache.get_mut(&segments) {
let (_, value) = &mut self.routes[*route_index];
return Some(Match { value, params: HashMap::default() });
};
for (index, (key, value)) in self.routes.iter_mut().enumerate() {
if let Some(Params(params)) = key.find(&segments.0) {
if params.is_empty() {
self.lookup_cache.insert(segments, RouteId(index));
}
return Some(Match { value, params });
};
}
None
}
}
#[cfg(test)]
mod lib_tests {
use super::*;
#[test]
fn test() {
let mut router = Router::default();
let nice = "nice".to_string();
router.at("/test", nice.clone());
router.at("/test/:var/:fdshj", nice.clone());
router.at("/test/:var", nice.clone());
router.at("/test2", nice.clone());
}
}