Add working part2/day2 solution

Switch to using new part1 solution as primary solution, make structs
and methods public for use in part2

Keep original solution as old1.rs
This commit is contained in:
Evie Litherland-Smith 2024-01-01 11:33:10 +00:00
parent 1137f8dc8b
commit 8c1d94d269
5 changed files with 266 additions and 214 deletions

View file

@ -1,8 +1,7 @@
mod part1; mod part1;
mod part1alt; mod part2;
fn main() { fn main() {
println!("Part1: {}", part1::main(include_str!("input.txt"))); println!("Part1: {}", part1::main(include_str!("input.txt")));
println!("Part1alt: {}", part1alt::main(include_str!("input.txt"))); println!("Part2: {}", part2::main(include_str!("input.txt")));
println!("Part2: TODO");
} }

156
day2/src/old1.rs Normal file
View file

@ -0,0 +1,156 @@
use regex::{Match, Regex};
pub struct Games {
games: Vec<Game>,
}
impl Games {
fn new(input: &str, bag: Contents) -> Self {
Self {
games: input
.lines()
.map(|m| Game::new(m.to_owned(), bag.clone()))
.collect(),
}
}
fn valid_games(&self) -> Vec<i32> {
self.games
.iter()
.filter(|m| m.valid())
.map(|m| m.id)
.collect()
}
fn valid_sum(&self) -> i32 {
self.valid_games().iter().sum()
}
}
#[derive(Debug, Clone)]
pub struct Game {
bag: Contents,
id: i32,
draws: Vec<Contents>,
}
impl Game {
fn new(line: String, bag: Contents) -> Self {
Self {
bag: bag.clone(),
id: line
.split(':')
.next()
.unwrap()
.split(' ')
.last()
.expect("Must have game ID")
.parse::<i32>()
.expect("Game ID must be integer value"),
draws: Game::get_draws(line.split(':').last().unwrap()),
}
}
fn get_draws(line: &str) -> Vec<Contents> {
line.split(':')
.next()
.unwrap()
.split(';')
.map(|m| -> Contents { Contents::new(m) })
.collect()
}
fn valid(&self) -> bool {
self.draws.iter().all(|m| self.bag.valid(m))
}
}
#[derive(Debug, Clone)]
pub struct Contents {
blue: i32,
green: i32,
red: i32,
}
impl Contents {
fn new(line: &str) -> Self {
Self {
red: Self::get_red(line),
blue: Self::get_blue(line),
green: Self::get_green(line),
}
}
fn valid(self: &Contents, hand: &Contents) -> bool {
self.red.ge(&hand.red) & self.blue.ge(&hand.blue) & self.green.ge(&hand.green)
}
fn get_value(slice: &str) -> i32 {
Regex::new(r"(\d)+")
.unwrap()
.find(slice)
.unwrap()
.as_str()
.parse()
.unwrap()
}
fn get_slice<'a>(line: &'a str, rstring: &'a str) -> Option<Match<'a>> {
Regex::new(rstring).unwrap().find(line)
}
fn get_red(line: &str) -> i32 {
let rstring = r"((\d)+ red)";
match Self::get_slice(line, rstring) {
Some(slice) => Self::get_value(slice.as_str()),
None => 0,
}
}
fn get_blue(line: &str) -> i32 {
let rstring = r"((\d)+ blue)";
match Self::get_slice(line, rstring) {
Some(slice) => Self::get_value(slice.as_str()),
None => 0,
}
}
fn get_green(line: &str) -> i32 {
let rstring = r"((\d)+ green)";
match Self::get_slice(line, rstring) {
Some(slice) => Self::get_value(slice.as_str()),
None => 0,
}
}
}
pub fn main(input: &str) -> i32 {
let bag = Contents {
red: 12,
green: 13,
blue: 14,
};
let games = Games::new(input, bag);
games.valid_sum()
}
#[cfg(test)]
mod tests {
use super::*;
const BAG: Contents = Contents {
red: 12,
green: 13,
blue: 14,
};
const INPUT: &str = "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";
#[test]
fn example() {
assert_eq!(main(INPUT), 8)
}
#[test]
fn by_line() {
assert_eq!(Games::new(INPUT, BAG).valid_games(), vec![1, 2, 5])
}
#[test]
fn solution() {
assert_eq!(main(include_str!("input.txt")), 2476)
}
}

View file

@ -1,152 +1,92 @@
use regex::{Match, Regex}; use regex::Regex;
struct Games { type Games = Vec<Game>;
games: Vec<Game>,
#[derive(Debug)]
pub struct Bag {
pub red: usize,
pub green: usize,
pub blue: usize,
} }
impl Games { #[derive(Debug)]
fn new(input: &str, bag: Contents) -> Self { pub struct Game {
Self { pub id: usize,
games: input pub rounds: Vec<Round>,
.lines()
.map(|m| Game::new(m.to_owned(), bag.clone()))
.collect(),
}
}
fn valid_games(&self) -> Vec<i32> {
self.games
.iter()
.filter(|m| m.valid())
.map(|m| m.id)
.collect()
}
fn valid_sum(&self) -> i32 {
self.valid_games().iter().sum()
}
}
#[derive(Debug, Clone)]
struct Game {
bag: Contents,
id: i32,
draws: Vec<Contents>,
} }
impl Game { impl Game {
fn new(line: String, bag: Contents) -> Self { pub fn new(input: &str) -> Self {
let (id, games) = input
.split_once(':')
.expect("All input lines follow this format");
Self { Self {
bag: bag.clone(), id: id.split(' ').last().unwrap().parse().unwrap(),
id: line rounds: games.split(';').map(Round::new).collect(),
.split(':')
.next()
.unwrap()
.split(' ')
.last()
.expect("Must have game ID")
.parse::<i32>()
.expect("Game ID must be integer value"),
draws: Game::get_draws(line.split(':').last().unwrap()),
} }
} }
fn get_draws(line: &str) -> Vec<Contents> { pub fn is_valid(&self, bag: &Bag) -> bool {
line.split(':') self.rounds.iter().all(|m| m.is_valid(bag))
.next()
.unwrap()
.split(';')
.map(|m| -> Contents { Contents::new(m) })
.collect()
}
fn valid(&self) -> bool {
self.draws.iter().all(|m| self.bag.valid(m))
} }
} }
#[derive(Debug, Clone)] #[derive(Debug)]
struct Contents { pub struct Round {
blue: i32, pub red: usize,
green: i32, pub green: usize,
red: i32, pub blue: usize,
}
impl Round {
pub fn new(input: &str) -> Self {
fn get_colour(rstring: &str, input: &str) -> usize {
match Regex::new(rstring).unwrap().find(input) {
Some(val) => val.as_str().split(' ').next().unwrap().parse().unwrap(),
None => 0,
}
} }
impl Contents {
fn new(line: &str) -> Self {
Self { Self {
red: Self::get_red(line), red: get_colour(r"\d+ (red)", input),
blue: Self::get_blue(line), green: get_colour(r"\d+ (green)", input),
green: Self::get_green(line), blue: get_colour(r"\d+ (blue)", input),
}
}
fn valid(self: &Contents, hand: &Contents) -> bool {
self.red.ge(&hand.red) & self.blue.ge(&hand.blue) & self.green.ge(&hand.green)
}
fn get_value(slice: &str) -> i32 {
Regex::new(r"(\d)+")
.unwrap()
.find(slice)
.unwrap()
.as_str()
.parse()
.unwrap()
}
fn get_slice<'a>(line: &'a str, rstring: &'a str) -> Option<Match<'a>> {
Regex::new(rstring).unwrap().find(line)
}
fn get_red(line: &str) -> i32 {
let rstring = r"((\d)+ red)";
match Self::get_slice(line, rstring) {
Some(slice) => Self::get_value(slice.as_str()),
None => 0,
}
}
fn get_blue(line: &str) -> i32 {
let rstring = r"((\d)+ blue)";
match Self::get_slice(line, rstring) {
Some(slice) => Self::get_value(slice.as_str()),
None => 0,
}
}
fn get_green(line: &str) -> i32 {
let rstring = r"((\d)+ green)";
match Self::get_slice(line, rstring) {
Some(slice) => Self::get_value(slice.as_str()),
None => 0,
}
} }
} }
pub fn main(input: &str) -> i32 { pub fn is_valid(&self, bag: &Bag) -> bool {
let bag = Contents { self.red <= bag.red && self.green <= bag.green && self.blue <= bag.blue
}
}
pub fn games(input: &str) -> Games {
input.lines().map(Game::new).collect()
}
pub fn main(input: &str) -> usize {
let bag = Bag {
red: 12, red: 12,
green: 13, green: 13,
blue: 14, blue: 14,
}; };
let games = Games::new(input, bag); games(input)
games.valid_sum() .iter()
.filter_map(|m| if m.is_valid(&bag) { Some(m.id) } else { None })
.sum()
} }
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::*; use super::*;
const BAG: Contents = Contents {
red: 12, #[test]
green: 13, fn example() {
blue: 14, let input = "Game 1: 3 blue, 4 red; 1 red, 2 green, 6 blue; 2 green
};
const INPUT: &str = "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 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 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 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"; Game 5: 6 red, 1 blue, 3 green; 2 blue, 1 red, 2 green";
assert_eq!(main(input), 8);
#[test]
fn example() {
assert_eq!(main(INPUT), 8)
}
#[test]
fn by_line() {
assert_eq!(Games::new(INPUT, BAG).valid_games(), vec![1, 2, 5])
} }
#[test] #[test]

View file

@ -1,96 +0,0 @@
use regex::Regex;
// type Round<'a> = &'a str;
// type Game<'a> = Vec<Round>;
#[derive(Debug)]
struct Bag {
red: usize,
green: usize,
blue: usize,
}
#[derive(Debug)]
struct Game {
id: usize,
rounds: Vec<Round>,
}
impl Game {
fn new(input: &str) -> Self {
let (id, games) = input
.split_once(':')
.expect("All input lines follow this format");
Self {
id: id.split(' ').last().unwrap().parse().unwrap(),
rounds: games.split(';').map(Round::new).collect(),
}
}
fn is_valid(&self, bag: &Bag) -> bool {
self.rounds.iter().all(|m| m.is_valid(bag))
}
}
#[derive(Debug)]
struct Round {
red: usize,
green: usize,
blue: usize,
}
impl Round {
fn new(input: &str) -> Self {
fn get_colour(rstring: &str, input: &str) -> usize {
match Regex::new(rstring).unwrap().find(input) {
Some(val) => val.as_str().split(' ').next().unwrap().parse().unwrap(),
None => 0,
}
}
Self {
red: get_colour(r"\d+ (red)", input),
green: get_colour(r"\d+ (green)", input),
blue: get_colour(r"\d+ (blue)", input),
}
}
fn is_valid(&self, bag: &Bag) -> bool {
self.red <= bag.red && self.green <= bag.green && self.blue <= bag.blue
}
}
pub fn main(input: &str) -> usize {
let bag = Bag {
red: 12,
green: 13,
blue: 14,
};
let games: Vec<Game> = input.lines().map(Game::new).collect();
// dbg!(&games);
// dbg!(&games.iter().map(|m| m.is_valid(&bag)));
games
.iter()
.filter_map(|m| if m.is_valid(&bag) { Some(m.id) } else { None })
.sum()
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn example() {
let input = "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";
assert_eq!(main(input), 8);
}
#[test]
fn solution() {
assert_eq!(main(include_str!("input.txt")), 2476)
}
}

53
day2/src/part2.rs Normal file
View file

@ -0,0 +1,53 @@
use crate::part1::{games, Bag, Game};
impl Game {
pub fn min_bag(&self) -> Bag {
Bag {
red: self.rounds.iter().map(|m| m.red).max().unwrap(),
green: self.rounds.iter().map(|m| m.green).max().unwrap(),
blue: self.rounds.iter().map(|m| m.blue).max().unwrap(),
}
}
pub fn product(&self) -> usize {
let bag = self.min_bag();
bag.red * bag.green * bag.blue
}
}
pub fn products(input: &str) -> Vec<usize> {
games(input)
.iter()
.map(Game::product)
.collect::<Vec<usize>>()
}
pub fn main(input: &str) -> usize {
products(input).iter().sum()
}
#[cfg(test)]
mod tests {
use super::*;
const INPUT: &str = "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";
#[test]
fn example() {
assert_eq!(main(INPUT), 2286);
}
#[test]
fn by_line() {
assert_eq!(products(INPUT), vec![48, 12, 1560, 630, 36])
}
#[test]
fn solution() {
assert_eq!(main(include_str!("input.txt")), 54911)
}
}