use std::convert::TryFrom; use aoc_runner_derive::{aoc, aoc_generator, aoc_lib}; #[aoc_generator(day1)] pub fn d1_input(input: &str) -> Vec { let mut entries: Vec = input.split('\n').map(|x| x.parse().unwrap()).collect(); entries.sort(); entries } #[aoc(day1, part1)] pub fn solve_d1_p1(input: &[u64]) -> u64 { for entry in input { let needed = 2020 - entry; if let Ok(idx) = input.binary_search(&needed) { return entry * input[idx]; } } panic!("not found"); } #[aoc(day1, part2)] pub fn solve_d1_p2(input: &[u64]) -> Option { for i in 0..input.len() - 2 { let entry1 = input[i]; for j in i + 1..input.len() - 1 { let entry2 = input[j]; if entry1 + entry2 > 2020 { break; } for k in j + 1..input.len() { let entry3 = input[k]; if entry1 + entry2 + entry3 == 2020 { return Some(entry1 * entry2 * entry3); } } } } return None; } struct PasswdEntry<'a> { lower_bound: usize, upper_bound: usize, policy_char: u8, passwd: &'a [u8], } impl<'a> PasswdEntry<'a> { fn parse(mut input: &'a str) -> Option { let lb_idx = input.find('-')?; let lower_bound: usize = (&input[..lb_idx]).parse().ok()?; input = &input[lb_idx + 1..]; let ub_idx = input.find(' ')?; let upper_bound: usize = (&input[..ub_idx]).parse().ok()?; input = &input[ub_idx + 1..]; let policy_char = input.as_bytes()[0]; let passwd = &input[3..].as_bytes(); Some(PasswdEntry { lower_bound, upper_bound, policy_char, passwd }) } } #[aoc(day2, part1)] pub fn solve_d2_p1(input: &str) -> usize { fn line_is_valid(line: &str) -> bool { let entry = PasswdEntry::parse(line).unwrap(); let count = entry.passwd.iter().filter(|&&b| b == entry.policy_char).count(); (count >= entry.lower_bound) && (count <= entry.upper_bound) } input.split('\n').filter(|x| line_is_valid(x)).count() } #[aoc(day2, part2)] pub fn solve_d2_p2(input: &str) -> usize { fn line_is_valid(line: &str) -> bool { let entry = PasswdEntry::parse(line).unwrap(); (entry.passwd[entry.lower_bound - 1] == entry.policy_char) ^ (entry.passwd[entry.upper_bound - 1] == entry.policy_char) } input.split('\n').filter(|x| line_is_valid(x)).count() } #[derive(Debug)] pub enum MapSquare { Open, Tree, } impl TryFrom for MapSquare { type Error = u8; fn try_from(b: u8) -> Result { Ok(match b { b'.' => MapSquare::Open, b'#' => MapSquare::Tree, unknown => return Err(unknown), }) } } #[aoc_generator(day3)] pub fn d3_input(input: &[u8]) -> Vec> { input .split(|&b| b == b'\n') .map(|line| line.iter().copied().map(|b| MapSquare::try_from(b).unwrap()).collect()) .collect() } #[aoc(day3, part1)] pub fn solve_d3_p1(input: &[Vec]) -> usize { input .iter() .skip(1) .enumerate() .map(|(steps_taken, grid_line)| { let x_coord = (steps_taken + 1) * 3; match grid_line.iter().cycle().nth(x_coord).unwrap() { MapSquare::Open => 0, MapSquare::Tree => 1, } }) .sum() } #[aoc(day3, part2)] pub fn solve_d3_p2(input: &[Vec]) -> usize { #[derive(Debug, Clone, Copy)] struct Step { x_step: usize, y_step: usize, } let steps = [ Step { x_step: 1, y_step: 1 }, Step { x_step: 3, y_step: 1 }, Step { x_step: 5, y_step: 1 }, Step { x_step: 7, y_step: 1 }, Step { x_step: 1, y_step: 2 }, ]; steps .iter() .copied() .map(|Step { x_step, y_step }| -> usize { input .iter() .skip(y_step) .step_by(y_step) .enumerate() .map(|(num_steps_taken, grid_line)| { let x_coord = (num_steps_taken + 1) * x_step; match grid_line.iter().cycle().nth(x_coord).unwrap() { MapSquare::Open => 0, MapSquare::Tree => 1, } }) .sum() }) .product() } aoc_lib! { year = 2020 }