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:
parent
1137f8dc8b
commit
8c1d94d269
|
@ -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
156
day2/src/old1.rs
Normal 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)
|
||||||
|
}
|
||||||
|
}
|
|
@ -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]
|
||||||
|
|
|
@ -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
53
day2/src/part2.rs
Normal 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)
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue