rrule 0.14.0

A pure Rust implementation of recurrence rules as defined in the iCalendar RFC.
Documentation
use std::{fmt::Display, str::FromStr};

use clap::Parser;
use rrule::RRuleSet;

/// Recurrence Rule parser and iterator
///
/// This program expects a Recurrence Rule (RRULE) as defined by the
/// [iCalendar (RFC-5545) specification](https://icalendar.org/RFC-Specifications/iCalendar-RFC-5545/).
///
/// Some `RRuleSet` examples of a valid iCalendar string:
///
/// - `DTSTART:20120201T093000Z\nRRULE:FREQ=YEARLY`
/// - `DTSTART:20120201T093000Z\nRRULE:FREQ=WEEKLY;INTERVAL=5;BYDAY=MO,FR`
#[derive(Parser, Debug)]
#[clap(
    name = "rrule",
    about = "A parser and iterator for recurrence rules as defined in the iCalendar RFC."
)]
struct Opts {
    /// Limits the amount of iteration
    /// If no limit is set, it will default to `100`.
    /// The maximum limit is `65535`.
    #[clap(short, long)]
    limit: Option<u16>,

    /// The `RRULE` string you want to iterator over.
    input: String,
}

fn main() -> Result<(), String> {
    let opts: Opts = Parser::parse();

    let limit = opts.limit.unwrap_or(100);
    let rrule_str = opts.input.replace("\\n", "\n");
    let rrule: RRuleSet = parse_rule(&rrule_str)?;
    let rrule = rrule.limit();
    let iter = rrule.into_iter();
    iterator_dates(iter, limit);

    Ok(())
}

fn parse_rule<R: FromStr>(rrule_str: &str) -> Result<R, String>
where
    <R as FromStr>::Err: Display,
{
    match rrule_str.parse() {
        Ok(rrule) => Ok(rrule),
        Err(err) => {
            eprintln!("Error: {}", err);
            Err("Input string needs to be a valid rrule string.".to_owned())
        }
    }
}

fn iterator_dates<T>(mut rule_iter: T, limit: u16)
where
    T: Iterator,
    <T as Iterator>::Item: Display,
{
    for _ in 0..limit {
        let next = rule_iter.next();
        if let Some(value) = next {
            println!("{}", value);
        }
    }
}