2021-04-17 22:39:22 +02:00
|
|
|
use diesel::Queryable;
|
2021-04-05 10:48:10 +02:00
|
|
|
use regex::Regex;
|
|
|
|
use rocket::http::RawStr;
|
|
|
|
use rocket::request::FromFormValue;
|
|
|
|
use serde::ser::{Serialize, SerializeStruct, Serializer};
|
|
|
|
use std::convert::TryFrom;
|
|
|
|
|
2021-04-12 21:33:56 +02:00
|
|
|
/// Represents a street in a given city
|
2021-04-17 22:39:22 +02:00
|
|
|
#[derive(Queryable)]
|
2021-04-05 10:48:10 +02:00
|
|
|
pub struct Street {
|
2021-04-08 22:53:29 +02:00
|
|
|
name: String,
|
|
|
|
city: String,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Street {
|
|
|
|
// This constructor just makes my life a bit easier during testing
|
|
|
|
#[cfg(test)]
|
|
|
|
fn new(name: String, city: String) -> Street {
|
|
|
|
Street {
|
|
|
|
name: name,
|
|
|
|
city: city,
|
|
|
|
}
|
|
|
|
}
|
2021-04-05 10:48:10 +02:00
|
|
|
}
|
|
|
|
|
2021-04-08 10:11:15 +02:00
|
|
|
impl From<&Street> for String {
|
|
|
|
fn from(street: &Street) -> String {
|
2021-04-05 10:48:10 +02:00
|
|
|
format!("{} ({})", street.name, street.city)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Serialize for Street {
|
|
|
|
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
|
|
|
where
|
|
|
|
S: Serializer,
|
|
|
|
{
|
|
|
|
let mut s = serializer.serialize_struct("Street", 2)?;
|
|
|
|
s.serialize_field("name", &self.name)?;
|
|
|
|
s.serialize_field("city", &self.city)?;
|
|
|
|
s.end()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-05 11:10:48 +02:00
|
|
|
impl TryFrom<&str> for Street {
|
2021-04-05 10:48:10 +02:00
|
|
|
type Error = ();
|
|
|
|
|
2021-04-05 11:10:48 +02:00
|
|
|
fn try_from(value: &str) -> Result<Self, Self::Error> {
|
2021-04-05 10:48:10 +02:00
|
|
|
if let Some(index) = value.find('(') {
|
|
|
|
Ok(Street {
|
|
|
|
name: (value[0..index - 1].trim()).to_string(),
|
|
|
|
city: (value[index + 1..value.len() - 1].trim()).to_string(),
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
Err(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'v> FromFormValue<'v> for Street {
|
|
|
|
type Error = &'v RawStr;
|
|
|
|
|
|
|
|
fn from_form_value(form_value: &'v RawStr) -> Result<Street, Self::Error> {
|
|
|
|
// This regex is pretty loose tbh, but not sure how I can make it more
|
|
|
|
// strict right now
|
|
|
|
let re = Regex::new(r"^(.+) \((.+)\)$").unwrap();
|
2021-04-08 22:53:29 +02:00
|
|
|
|
2021-04-05 10:48:10 +02:00
|
|
|
match re.captures(&form_value.url_decode_lossy()) {
|
|
|
|
None => Err(form_value),
|
|
|
|
Some(caps) => Ok(Street {
|
|
|
|
name: String::from(caps.get(1).unwrap().as_str()),
|
|
|
|
city: String::from(caps.get(2).unwrap().as_str()),
|
|
|
|
}),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-04-08 22:53:29 +02:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
2021-04-12 21:33:56 +02:00
|
|
|
/// Tests the conversion to string
|
2021-04-08 22:53:29 +02:00
|
|
|
#[test]
|
|
|
|
fn test_to_string() {
|
|
|
|
let street = Street::new(String::from("testname"), String::from("city"));
|
|
|
|
|
|
|
|
assert_eq!(String::from("testname (city)"), String::from(&street));
|
|
|
|
}
|
|
|
|
}
|