Opening hours completed. Leptos upgraded to 0.5.2.

This commit is contained in:
2023-11-11 16:27:40 +01:00
parent e7af2d402d
commit 1de6b74665
13 changed files with 469 additions and 77 deletions
+176 -9
View File
@@ -1,9 +1,14 @@
//use chrono::{NaiveDate, NaiveTime, Weekday};
//use rust_decimal::Decimal;
#![allow(unused_variables)]
use chrono::{NaiveTime, Weekday};
use lazy_static::lazy_static;
use regex::Regex;
use serde::{Deserialize, Serialize};
//use uuid::Uuid;
use validator::Validate;
use validator::{Validate, ValidationError};
use crate::error::AppError;
#[derive(Serialize, Deserialize, Clone, Debug)]
pub enum ApiResponse<T> {
@@ -114,6 +119,176 @@ impl PwdChange {
}
}
#[derive(Clone, Serialize, Deserialize, Debug, PartialEq, Eq, Default)]
#[cfg_attr(feature = "ssr", derive(sqlx::FromRow))]
pub struct OpeningHour {
id: i32,
pub day: i32,
pub from: NaiveTime,
pub to: NaiveTime,
pub discount: Option<i32>
}
impl OpeningHour {
pub fn id(&self) -> i32 {
self.id
}
}
#[derive(Serialize, Deserialize, Clone, Debug)]
pub struct DayHours(pub Vec<DayHour>);
impl DayHours {
pub fn try_new(hours: &str) -> Result<Self, AppError> {
if hours.is_empty() {
return Ok(Self(Vec::new()))
}
let times = hours.split(",")
.map(|h| h.trim())
.map(|h| DayHour::try_from(h))
.collect::<Vec<_>>();
if times.contains(&Err(AppError::HourParseError)) {
return Err(AppError::HourParseError)
}
Ok(Self(times.into_iter().map(|h| h.unwrap()).collect()))
}
pub fn hours(&self) -> &Vec<DayHour> {
&self.0
}
}
impl ToString for DayHours {
fn to_string(&self) -> String {
if self.0.is_empty() {
return "".to_string()
}
self.0.iter().map(|h| {
let discount = if let Some(d) = h.discount() {
format!(" ({})", d).to_string()
} else { "".to_string() };
format!("{} - {}{}", h.from().format("%H:%M"), h.to().format("%H:%M"), discount).to_string()})
.collect::<Vec<String>>()
.join(", ")
}
}
#[derive(Eq, PartialEq, Serialize, Deserialize, Clone, Debug)]
pub struct DayHour {
from: NaiveTime,
to: NaiveTime,
discount: Option<i32>
}
impl DayHour {
pub fn new(from: NaiveTime, to: NaiveTime, discount: Option<i32>) -> Self {
Self { from, to, discount }
}
pub fn from(&self) -> NaiveTime {
self.from
}
pub fn to(&self) -> NaiveTime {
self.to
}
pub fn discount(&self) -> Option<i32> {
self.discount
}
}
impl TryFrom<String> for DayHour {
type Error = AppError;
fn try_from(value: String) -> Result<Self, Self::Error> {
value.as_str().try_into()
}
}
impl TryFrom<&str> for DayHour {
type Error = AppError;
fn try_from(value: &str) -> Result<Self, Self::Error> {
if value.is_empty() {
return Err(AppError::HourParseError)
}
let times = value.split("-")
.map(|t| t.trim()).collect::<Vec<_>>();
if times.len() != 2 {
return Err(AppError::HourParseError)
}
let from = NaiveTime::parse_from_str(times.get(0).unwrap(), "%H:%M");
let to = NaiveTime::parse_from_str(times.get(1).unwrap(), "%H:%M");
if from.is_err() || to.is_err() {
return Err(AppError::HourParseError)
}
Ok(DayHour {
from: from.unwrap(),
to: to.unwrap(),
discount: None
})
}
}
fn validate_hours(value: &WeekHours) -> Result<(), ValidationError> {
if value.hours().is_empty() {
return Ok(())
}
if let Ok(h) = DayHours::try_new(value.hours()) {
for hr in h.hours() {
if hr.from() >= hr.to() { return Err(ValidationError::new("TO_BEFORE_FROM")) }
}
Ok(())
} else {
Ok(())
}
}
lazy_static! {
static ref RE_HOURS: Regex = Regex::new(r"^$|(^\d{2}:\d{2} ?- ?\d{2}:\d{2} ?(\(\d+\))?,? ?)+").unwrap();
}
#[derive(Clone, Serialize, Deserialize, Debug, Validate)]
#[validate(schema(function = "validate_hours", message = "Time 'to' must be after time 'from'"))]
pub struct WeekHours {
day: Weekday,
#[validate(regex(path = "RE_HOURS", message = "Hours must be in HH:MM - HH:MM format"))]
hours: String
}
impl Default for WeekHours {
fn default() -> Self {
Self {
day: Weekday::Mon,
hours: String::default()
}
}
}
impl WeekHours {
pub fn new(day: Weekday, hours: Vec<DayHour>) -> Self {
Self {
day,
hours: DayHours(hours).to_string()
}
}
pub fn day(&self) -> Weekday {
self.day
}
pub fn hours(&self) -> &str {
&self.hours
}
}
/*pub struct Property {
id: u16,
name: String,
@@ -135,14 +310,6 @@ pub struct Message {
text: String,
}
pub struct OpeningHour {
id: u16,
day: Weekday,
from: NaiveTime,
to: NaiveTime,
discount: u8
}
pub struct Customer {
id: u128,
full_name: String,
+1
View File
@@ -4,6 +4,7 @@ pub mod data;
pub mod company;
pub mod user;
pub mod auth_middleware;
pub mod opening_hours;
#[macro_export]
macro_rules! perm_check {
+69
View File
@@ -0,0 +1,69 @@
use std::collections::HashMap;
use chrono::Weekday;
use leptos::*;
use validator::Validate;
use crate::backend::data::{ApiResponse, DayHour, WeekHours};
use crate::components::data_form::ForValidation;
#[server]
pub async fn get_hours() -> Result<HashMap<Weekday, Vec<DayHour>>, ServerFnError> {
use crate::backend::get_pool;
use crate::backend::data::OpeningHour;
let pool = get_pool().await?;
let hours = sqlx::query_as::<_, OpeningHour>("SELECT * FROM opening_hour").fetch_all(&pool).await?;
let mut ret: HashMap<Weekday, Vec<DayHour>> = hours.into_iter().fold(HashMap::new(), |mut map, v| {
map.entry(Weekday::try_from(v.day as u8).unwrap_or(Weekday::Mon))
.and_modify(|h| h.push(DayHour::new(v.from, v.to, v.discount)))
.or_insert(vec![DayHour::new(v.from, v.to, v.discount)]);
map
});
for d in 0..6 {
if let None = ret.get(&Weekday::try_from(d as u8).unwrap()) {
ret.insert(Weekday::try_from(d as u8).unwrap(), Vec::new());
}
}
Ok(ret)
}
#[server]
pub async fn update_hours(hours: WeekHours) -> Result<ApiResponse<()>, ServerFnError> {
use crate::perm_check;
use crate::backend::get_pool;
use crate::backend::data::DayHours;
perm_check!(is_admin);
let hr = DayHours::try_new(hours.hours())?;
let pool = get_pool().await?;
let day = hours.day();
let mut tx = pool.begin().await?;
sqlx::query("DELETE FROM opening_hour WHERE day = $1")
.bind(day.num_days_from_monday() as i32)
.execute(&mut *tx)
.await?;
for h in hr.hours() {
sqlx::query(r#"INSERT INTO opening_hour(day, "from", "to", discount) VALUES($1, $2, $3, $4)"#)
.bind(day.num_days_from_monday() as i32)
.bind(h.from())
.bind(h.to())
.bind(h.discount())
.execute(&mut *tx)
.await?;
}
tx.commit().await?;
Ok(ApiResponse::Data(()))
}
impl ForValidation for UpdateHours {
fn entity(&self) -> &dyn Validate {
&self.hours
}
}