use crate::time::calendars::Calendar;
use chrono::NaiveDate;
use serde::{Deserialize, Serialize};
#[derive(Deserialize, Serialize, Debug)]
pub enum FranceMarket {
Settlement,
Exchange,
}
#[derive(Deserialize, Serialize, Default, Debug)]
pub struct France {
pub market: Option<FranceMarket>,
}
impl France {
fn settlement_is_business_day(&self, date: NaiveDate) -> bool {
let (d, _, m, y, dd) = self.naive_date_to_dkmy(date);
let em = self.easter_monday(y);
if self.is_weekend(date)
|| (d == 1 && m == 1)
|| (dd == em)
|| (d == 1 && m == 5)
|| (d == 8 && m == 5)
|| (d == 10 && m == 5)
|| (d == 21 && m == 5)
|| (d == 14 && m == 7)
|| (d == 15 && m == 8)
|| (d == 1 && m == 11)
|| (d == 11 && m == 11)
|| (d == 25 && m == 12)
{
return false;
}
true
}
fn exchange_is_business_day(&self, date: NaiveDate) -> bool {
let (d, _, m, y, dd) = self.naive_date_to_dkmy(date);
let em = self.easter_monday(y);
if self.is_weekend(date)
|| (d == 1 && m == 1)
|| (dd == em-3)
|| (dd == em)
|| (d == 1 && m == 5)
|| (d == 24 && m == 12)
|| (d == 25 && m == 12)
|| (d == 26 && m == 12)
|| (d == 31 && m == 12)
{
false
} else {
true
}
}
}
#[typetag::serde]
impl Calendar for France {
fn is_business_day(&self, date: NaiveDate) -> bool {
match self.market {
Some(FranceMarket::Settlement) => self.settlement_is_business_day(date),
Some(FranceMarket::Exchange) => self.exchange_is_business_day(date),
None => self.settlement_is_business_day(date),
}
}
}
#[cfg(test)]
mod tests {
use super::France;
use crate::time::calendars::Calendar;
use chrono::{Duration, NaiveDate};
#[test]
fn test_france_holiday() {
let expected_results_for_2023 = vec![
false, true, true, true, true, true, false, false, true, true, true, true, true, false,
false, true, true, true, true, true, false, false, true, true, true, true, true, false,
false, true, true, true, true, true, false, false, true, true, true, true, true, false,
false, true, true, true, true, true, false, false, true, true, true, true, true, false,
false, true, true, true, true, true, false, false, true, true, true, true, true, false,
false, true, true, true, true, true, false, false, true, true, true, true, true, false,
false, true, true, true, true, true, false, false, true, true, true, true, true, false,
false, false, true, true, true, true, false, false, true, true, true, true, true,
false, false, true, true, true, true, true, false, false, false, true, true, true,
true, false, false, false, true, false, true, true, false, false, true, true, true,
true, true, false, false, true, true, true, true, true, false, false, true, true, true,
true, true, false, false, true, true, true, true, true, false, false, true, true, true,
true, true, false, false, true, true, true, true, true, false, false, true, true, true,
true, true, false, false, true, true, true, true, true, false, false, true, true, true,
true, false, false, false, true, true, true, true, true, false, false, true, true,
true, true, true, false, false, true, true, true, true, true, false, false, true, true,
true, true, true, false, false, true, false, true, true, true, false, false, true,
true, true, true, true, false, false, true, true, true, true, true, false, false, true,
true, true, true, true, false, false, true, true, true, true, true, false, false, true,
true, true, true, true, false, false, true, true, true, true, true, false, false, true,
true, true, true, true, false, false, true, true, true, true, true, false, false, true,
true, true, true, true, false, false, true, true, true, true, true, false, false, true,
true, false, true, true, false, false, true, true, true, true, true, false, false,
true, true, true, true, true, false, false, true, true, true, true, true, false, false,
true, true, true, true, true, false, false, true, true, true, true, true, false, false,
true, true, true, true, true, false, false, true, true, true, true, true, false, false,
false, true, true, true, true, false, false,
];
let first_date = NaiveDate::from_ymd_opt(2023, 1, 1).unwrap();
for n in 0i32..365 {
let target_date = first_date + Duration::try_days(n as i64).unwrap();
let expected = expected_results_for_2023[n as usize];
assert_eq!(France::default().is_business_day(target_date), expected);
}
}
}