2022-10-17 11:19:08 +02:00
|
|
|
// This file contains template helpers.
|
|
|
|
// Prefer `./helpers.rs` if you want to extract code from your solutions.
|
2021-12-29 14:12:01 +01:00
|
|
|
use std::env;
|
|
|
|
use std::fs;
|
|
|
|
|
2022-10-17 11:19:08 +02:00
|
|
|
pub mod helpers;
|
|
|
|
|
2021-12-29 19:14:40 +01:00
|
|
|
pub const ANSI_ITALIC: &str = "\x1b[3m";
|
|
|
|
pub const ANSI_BOLD: &str = "\x1b[1m";
|
|
|
|
pub const ANSI_RESET: &str = "\x1b[0m";
|
2021-12-29 14:36:48 +01:00
|
|
|
|
|
|
|
#[macro_export]
|
2021-12-29 19:14:40 +01:00
|
|
|
macro_rules! solve {
|
2022-10-29 17:04:13 +02:00
|
|
|
($part:expr, $solver:ident, $input:expr) => {{
|
2021-12-29 14:36:48 +01:00
|
|
|
use aoc::{ANSI_BOLD, ANSI_ITALIC, ANSI_RESET};
|
|
|
|
use std::fmt::Display;
|
|
|
|
use std::time::Instant;
|
|
|
|
|
2022-10-29 17:04:13 +02:00
|
|
|
fn print_result<T: Display>(func: impl FnOnce(&str) -> Option<T>, input: &str) {
|
2021-12-29 14:36:48 +01:00
|
|
|
let timer = Instant::now();
|
|
|
|
let result = func(input);
|
2021-12-30 11:34:27 +01:00
|
|
|
let elapsed = timer.elapsed();
|
2022-10-29 17:04:13 +02:00
|
|
|
match result {
|
|
|
|
Some(result) => {
|
|
|
|
println!(
|
|
|
|
"{} {}(elapsed: {:.2?}){}",
|
|
|
|
result, ANSI_ITALIC, elapsed, ANSI_RESET
|
|
|
|
);
|
|
|
|
},
|
|
|
|
None => {
|
|
|
|
println!("not solved.")
|
|
|
|
}
|
|
|
|
}
|
2021-12-29 14:36:48 +01:00
|
|
|
}
|
|
|
|
|
2022-10-29 17:04:13 +02:00
|
|
|
println!("🎄 {}Part {}{} 🎄", ANSI_BOLD, $part, ANSI_RESET);
|
|
|
|
print_result($solver, $input);
|
2021-12-29 14:36:48 +01:00
|
|
|
}};
|
|
|
|
}
|
|
|
|
|
2021-12-29 14:12:01 +01:00
|
|
|
pub fn read_file(folder: &str, day: u8) -> String {
|
|
|
|
let cwd = env::current_dir().unwrap();
|
|
|
|
|
2021-12-29 14:36:48 +01:00
|
|
|
let filepath = cwd.join("src").join(folder).join(format!("{:02}.txt", day));
|
2021-12-29 14:12:01 +01:00
|
|
|
|
|
|
|
let f = fs::read_to_string(filepath);
|
|
|
|
f.expect("could not open input file")
|
|
|
|
}
|
2021-12-29 19:14:40 +01:00
|
|
|
|
|
|
|
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 {
|
2021-12-31 13:05:23 +01:00
|
|
|
let timing = l.split("(elapsed: ").last().unwrap();
|
|
|
|
// use `contains` istd. of `ends_with`: string may contain ANSI escape sequences.
|
2022-10-17 11:19:08 +02:00
|
|
|
// for possible time formats, see: https://github.com/rust-lang/rust/blob/1.64.0/library/core/src/time.rs#L1176-L1200
|
2021-12-31 13:05:23 +01:00
|
|
|
if timing.contains("ns)") {
|
2021-12-29 19:14:40 +01:00
|
|
|
acc // range below rounding precision.
|
2021-12-31 13:05:23 +01:00
|
|
|
} else if timing.contains("µs)") {
|
2021-12-29 19:14:40 +01:00
|
|
|
acc + parse_time(timing, "µs") / 1000_f64
|
2021-12-31 13:05:23 +01:00
|
|
|
} else if timing.contains("ms)") {
|
2021-12-29 19:14:40 +01:00
|
|
|
acc + parse_time(timing, "ms")
|
2021-12-31 13:05:23 +01:00
|
|
|
} else if timing.contains("s)") {
|
2021-12-29 19:14:40 +01:00
|
|
|
acc + parse_time(timing, "s") * 1000_f64
|
|
|
|
} else {
|
|
|
|
acc
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-10-17 11:19:08 +02:00
|
|
|
/// 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
|
|
|
|
);
|
|
|
|
}};
|
|
|
|
}
|
|
|
|
|
2021-12-29 19:14:40 +01:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_parse_exec_time() {
|
|
|
|
assert_approx_eq!(
|
2021-12-31 13:05:23 +01:00
|
|
|
parse_exec_time(&format!(
|
|
|
|
"🎄 Part 1 🎄\n0 (elapsed: 74.13ns){}\n🎄 Part 2 🎄\n0 (elapsed: 50.00ns){}",
|
|
|
|
ANSI_RESET, ANSI_RESET
|
|
|
|
)),
|
2021-12-29 19:14:40 +01:00
|
|
|
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
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|