advent-of-code/2023/rust/src/day2.rs

124 lines
3.2 KiB
Rust
Raw Normal View History

2023-12-02 15:07:19 -06:00
mod prelude;
2023-12-02 15:50:21 -06:00
extern crate nom;
use crate::prelude::*;
2023-12-02 15:07:19 -06:00
use nom::{
branch::alt,
2023-12-02 15:50:21 -06:00
bytes::complete::{tag, take_while},
2023-12-02 15:07:19 -06:00
combinator::{map_res, value},
multi::separated_list0,
IResult,
};
2023-12-02 15:50:21 -06:00
use std::cmp::max;
2023-12-02 15:07:19 -06:00
fn main() {
Day2::show(day_input(2), day_input(2))
}
#[derive(Debug, PartialEq, Clone)]
enum Reveal {
Red(usize),
Green(usize),
Blue(usize),
}
#[derive(Debug, PartialEq)]
struct Game {
id: usize,
2023-12-02 15:50:21 -06:00
reveals: Vec<Reveal>,
2023-12-02 15:18:20 -06:00
}
2023-12-02 15:50:21 -06:00
// parser
2023-12-02 15:07:19 -06:00
fn int(input: &str) -> IResult<&str, usize> {
2023-12-02 15:50:21 -06:00
map_res(take_while(char::is_numeric), str::parse)(input)
2023-12-02 15:07:19 -06:00
}
fn reveal(input: &str) -> IResult<&str, Reveal> {
let (input, num) = int(input)?;
alt((
value(Reveal::Red(num), tag(" red")),
value(Reveal::Green(num), tag(" green")),
value(Reveal::Blue(num), tag(" blue")),
))(input)
}
fn game(input: &str) -> IResult<&str, Game> {
let (input, id) = int(tag("Game ")(input)?.0)?;
2023-12-02 15:50:21 -06:00
let (input, reveals) =
separated_list0(alt((tag(", "), tag("; "))), reveal)(tag(": ")(input)?.0)?;
Ok((input, Game { id, reveals }))
}
2023-12-02 15:07:19 -06:00
2023-12-02 15:50:21 -06:00
impl Game {
fn possible_power(&self) -> usize {
// TODO: probably the last place to cleanup nicely
let mut factors: [usize; 3] = [0, 0, 0];
for r in &self.reveals {
match r {
Reveal::Red(n) => factors[0] = max(*n, factors[0]),
Reveal::Green(n) => factors[1] = max(*n, factors[1]),
Reveal::Blue(n) => factors[2] = max(*n, factors[2]),
}
}
factors.iter().fold(1, |a, b| a * b)
}
2023-12-02 15:07:19 -06:00
}
struct Day2 {}
impl AoCSolution for Day2 {
type Input = String;
2023-12-02 15:50:21 -06:00
type Solution = usize;
2023-12-02 15:07:19 -06:00
fn part1(input: Self::Input) -> Self::Solution {
2023-12-02 15:50:21 -06:00
let valid = |r: &Reveal| match r {
Reveal::Red(n) => *n <= 12,
Reveal::Green(n) => *n <= 13,
Reveal::Blue(n) => *n <= 14,
2023-12-02 15:07:19 -06:00
};
input
.lines()
.map(|s| game(s).unwrap().1)
2023-12-02 15:50:21 -06:00
.filter(|g| g.reveals.iter().all(valid))
2023-12-02 15:07:19 -06:00
.map(|g| g.id)
.sum()
}
fn part2(input: Self::Input) -> Self::Solution {
2023-12-02 15:18:20 -06:00
input
.lines()
2023-12-02 15:50:21 -06:00
.map(|s| game(s).unwrap().1.possible_power())
2023-12-02 15:18:20 -06:00
.sum()
2023-12-02 15:07:19 -06:00
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test() {
assert_eq!(
Day2::part1(
r#"Game 1: 3 blue, 4 red; 1 red, 2 green, 6 blue; 2 green
Game 2: 1 blue, 2 green; 3 green, 4 blue, 1 red; 1 green, 1 blue
Game 3: 8 green, 6 blue, 20 red; 5 blue, 4 red, 13 green; 5 green, 1 red
Game 4: 1 green, 3 red, 6 blue; 3 green, 6 red; 3 green, 15 blue, 14 red
Game 5: 6 red, 1 blue, 3 green; 2 blue, 1 red, 2 green"#
.into()
),
8
);
2023-12-02 15:18:20 -06:00
assert_eq!(
Day2::part2(
r#"Game 1: 3 blue, 4 red; 1 red, 2 green, 6 blue; 2 green
Game 2: 1 blue, 2 green; 3 green, 4 blue, 1 red; 1 green, 1 blue
Game 3: 8 green, 6 blue, 20 red; 5 blue, 4 red, 13 green; 5 green, 1 red
Game 4: 1 green, 3 red, 6 blue; 3 green, 6 red; 3 green, 15 blue, 14 red
Game 5: 6 red, 1 blue, 3 green; 2 blue, 1 red, 2 green"#
.into()
),
2286
);
2023-12-02 15:07:19 -06:00
}
}