/* * This file contains template code. * There is no need to edit this file unless you want to change template functionality. * Prefer `./helpers.rs` if you want to extract code from your solutions. */ use std::env; use std::fs; pub mod helpers; pub const ANSI_ITALIC: &str = "\x1b[3m"; pub const ANSI_BOLD: &str = "\x1b[1m"; pub const ANSI_RESET: &str = "\x1b[0m"; #[macro_export] macro_rules! solve { ($part:expr, $solver:ident, $input:expr) => {{ use advent_of_code::{ANSI_BOLD, ANSI_ITALIC, ANSI_RESET}; use std::fmt::Display; use std::time::Instant; fn print_result(func: impl FnOnce(&str) -> Option, input: &str) { let timer = Instant::now(); let result = func(input); let elapsed = timer.elapsed(); match result { Some(result) => { println!( "{} {}(elapsed: {:.2?}){}", result, ANSI_ITALIC, elapsed, ANSI_RESET ); } None => { println!("not solved.") } } } println!("๐ŸŽ„ {}Part {}{} ๐ŸŽ„", ANSI_BOLD, $part, ANSI_RESET); print_result($solver, $input); }}; } pub fn read_file(folder: &str, day: u8) -> String { let cwd = env::current_dir().unwrap(); let filepath = cwd.join("src").join(folder).join(format!("{:02}.txt", day)); let f = fs::read_to_string(filepath); f.expect("could not open input file") } fn parse_time(val: &str, postfix: &str) -> f64 { val.split(postfix).next().unwrap().parse().unwrap() } pub fn parse_exec_time(output: &str) -> f64 { output.lines().fold(0_f64, |acc, l| { if !l.contains("elapsed:") { acc } else { let timing = l.split("(elapsed: ").last().unwrap(); // use `contains` istd. of `ends_with`: string may contain ANSI escape sequences. // for possible time formats, see: https://github.com/rust-lang/rust/blob/1.64.0/library/core/src/time.rs#L1176-L1200 if timing.contains("ns)") { acc // range below rounding precision. } else if timing.contains("ยตs)") { acc + parse_time(timing, "ยตs") / 1000_f64 } else if timing.contains("ms)") { acc + parse_time(timing, "ms") } else if timing.contains("s)") { acc + parse_time(timing, "s") * 1000_f64 } else { acc } } }) } /// copied from: https://github.com/rust-lang/rust/blob/1.64.0/library/std/src/macros.rs#L328-L333 #[cfg(test)] macro_rules! assert_approx_eq { ($a:expr, $b:expr) => {{ let (a, b) = (&$a, &$b); assert!( (*a - *b).abs() < 1.0e-6, "{} is not approximately equal to {}", *a, *b ); }}; } #[cfg(test)] mod tests { use super::*; #[test] fn test_parse_exec_time() { assert_approx_eq!( parse_exec_time(&format!( "๐ŸŽ„ Part 1 ๐ŸŽ„\n0 (elapsed: 74.13ns){}\n๐ŸŽ„ Part 2 ๐ŸŽ„\n0 (elapsed: 50.00ns){}", ANSI_RESET, ANSI_RESET )), 0_f64 ); assert_approx_eq!( parse_exec_time("๐ŸŽ„ Part 1 ๐ŸŽ„\n0 (elapsed: 755ยตs)\n๐ŸŽ„ Part 2 ๐ŸŽ„\n0 (elapsed: 700ยตs)"), 1.455_f64 ); assert_approx_eq!( parse_exec_time("๐ŸŽ„ Part 1 ๐ŸŽ„\n0 (elapsed: 70ยตs)\n๐ŸŽ„ Part 2 ๐ŸŽ„\n0 (elapsed: 1.45ms)"), 1.52_f64 ); assert_approx_eq!( parse_exec_time( "๐ŸŽ„ Part 1 ๐ŸŽ„\n0 (elapsed: 10.3s)\n๐ŸŽ„ Part 2 ๐ŸŽ„\n0 (elapsed: 100.50ms)" ), 10400.50_f64 ); } }