use std::convert::TryFrom; use std::io::Write; use clap::ArgMatches; use chrono::{DateTime, Utc, Local, Duration, Weekday, Datelike}; use regex::Regex; use crate::error::{Result, Error}; use crate::database::Database; use crate::formatters::Formatter; use crate::config::Config; use crate::regex::parse_regex; use crate::timeparse::parse_time; use super::{Command, display::{Sheet, entries_for_display}}; /// Given a local datetime, returns the time of the previous monday's start fn prev_monday(now: DateTime) -> DateTime { let on_monday = match now.weekday() { Weekday::Mon => now, Weekday::Tue => now - Duration::days(1), Weekday::Wed => now - Duration::days(2), Weekday::Thu => now - Duration::days(3), Weekday::Fri => now - Duration::days(4), Weekday::Sat => now - Duration::days(5), Weekday::Sun => now - Duration::days(6), }; on_monday.date().and_hms(0, 0, 0).with_timezone(&Utc) } #[derive(Default)] pub struct Args { ids: bool, end: Option>, format: Formatter, grep: Option, sheet: Option, } impl<'a> TryFrom<&'a ArgMatches<'a>> for Args { type Error = Error; fn try_from(matches: &'a ArgMatches) -> Result { Ok(Args { ids: matches.is_present("ids"), end: matches.value_of("end").map(|s| parse_time(s)).transpose()?, format: matches.value_of("format").unwrap().parse()?, grep: matches.value_of("grep").map(parse_regex).transpose()?, sheet: matches.value_of("sheet").map(|s| s.parse()).transpose()?, }) } } pub struct WeekCommand { } impl<'a> Command<'a> for WeekCommand { type Args = Args; fn handle(args: Self::Args, db: &mut D, out: &mut O, err: &mut E, _config: &Config) -> Result<()> where D: Database, O: Write, E: Write, { let start = prev_monday(Local::now()); entries_for_display(Some(start), args.end, args.sheet, db, out, err, args.format, args.ids, args.grep) } } #[cfg(test)] mod tests { use super::*; #[test] fn use_non_default_week_start() { unimplemented!() } }