use crate::time::calendars::Calendar;
use chrono::{NaiveDate, Weekday};
use serde::{Deserialize, Serialize};
#[derive(Deserialize, Serialize, Debug)]
pub enum AustraliaMarket {
Settlement,
ASX,
}
#[derive(Deserialize, Serialize, Default, Debug)]
pub struct Australia {
pub market: Option<AustraliaMarket>,
}
impl Australia {
fn shared_holiday(&self, date: NaiveDate) -> bool {
let (d, w, m, y, dd) = self.naive_date_to_dkmy(date);
let em = self.easter_monday(y);
if self.is_weekend(date)
|| ((d == 1 || ((d == 2 || d == 3) && w == Weekday::Mon)) && m == 1)
|| ((d == 26 || ((d == 27 || d == 28) && w == Weekday::Mon)) &&
m == 1)
|| (dd == em-3)
|| (dd == em)
|| (d == 25 && m == 4)
|| ((d > 7 && d <= 14) && w == Weekday::Mon && m == 6)
|| ((d == 25 || (d == 27 && (w == Weekday::Mon || w == Weekday::Tue)))
&& m == 12)
|| ((d == 26 || (d == 28 && (w == Weekday::Mon || w == Weekday::Tue)))
&& m == 12)
|| (d == 22 && m == 9 && y == 2022)
{
true
} else {
false
}
}
fn extra_holiday(&self, date: NaiveDate) -> bool {
let (d, w, m, _, _) = self.naive_date_to_dkmy(date);
(m == 10 || m == 8) && w == Weekday::Mon && d <= 7
}
fn settlement_is_business_day(&self, date: NaiveDate) -> bool {
!(self.shared_holiday(date) | self.extra_holiday(date))
}
fn asx_is_business_day(&self, date: NaiveDate) -> bool {
!self.shared_holiday(date)
}
}
#[typetag::serde]
impl Calendar for Australia {
fn is_business_day(&self, date: NaiveDate) -> bool {
match self.market {
Some(AustraliaMarket::Settlement) => self.settlement_is_business_day(date),
Some(AustraliaMarket::ASX) => self.asx_is_business_day(date),
None => self.settlement_is_business_day(date),
}
}
}
#[cfg(test)]
mod tests {
use super::Australia;
use super::AustraliaMarket;
use super::Calendar;
use chrono::{Duration, NaiveDate};
#[test]
fn test_australia_settlement_holiday() {
let expected_results_for_2023 = vec![
false, false, true, true, true, true, false, false, true, true, true, true, true,
false, false, true, true, true, true, true, false, false, true, true, true, false,
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,
false, false, false, false, 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, false,
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, 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, 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, false, 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!(
Australia {
market: Some(AustraliaMarket::Settlement)
}
.is_business_day(target_date),
expected
);
}
}
}