1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89
use std::collections::HashMap; pub trait Entity: Clone { fn from(tuple: &crate::Tuple<'_>) -> Self; fn get(&self, field: &str) -> Option<&dyn crate::ToSql>; } impl<T: crate::ToSql + crate::FromSql + Clone> Entity for T { fn from(tuple: &crate::Tuple<'_>) -> T { tuple.nth(0) } fn get(&self, _: &str) -> Option<&dyn crate::ToSql> { Some(self) } } impl< T: crate::FromSql + crate::ToSql + Clone, S: std::hash::BuildHasher + Default + Clone, > Entity for HashMap<String, T, S> { fn from(tuple: &crate::Tuple<'_>) -> Self { let mut hashmap = HashMap::default(); for x in 0..tuple.len() { let name = tuple.field_name(x).unwrap(); let value = tuple.nth(x); hashmap.insert(name, value); } hashmap } fn get(&self, field: &str) -> Option<&dyn crate::ToSql> { match self.get(field) { Some(value) => Some(value), None => None, } } } impl< T: crate::FromSql + crate::ToSql + Clone, S: std::hash::BuildHasher + Default + Clone, > Entity for HashMap<usize, T, S> { fn from(tuple: &crate::Tuple<'_>) -> Self { let mut hashmap = HashMap::default(); for x in 0..tuple.len() { let value = tuple.nth(x); hashmap.insert(x, value); } hashmap } fn get(&self, field: &str) -> Option<&dyn crate::ToSql> { match self.get(&field.parse::<usize>().unwrap()) { Some(value) => Some(value), None => None, } } } #[cfg(test)] mod test { use std::collections::HashMap; #[test] fn hashmap_str_from_sql() { let elephantry = crate::test::new_conn(); let results: Vec<HashMap<String, i32>> = elephantry.query("SELECT 1 as n", &[]).unwrap().collect(); assert_eq!(results[0].get("n"), Some(&1)); } #[test] fn hashmap_usize_from_sql() { let elephantry = crate::test::new_conn(); let results: Vec<HashMap<usize, i32>> = elephantry.query("SELECT 1 as n", &[]).unwrap().collect(); assert_eq!(results[0].get(&0), Some(&1)); } }