day01 finished

This commit is contained in:
Ruediger Ludwig 2023-01-26 21:52:09 +01:00
parent 284f099d3e
commit 68fefd064a
9 changed files with 2487 additions and 164 deletions

View file

@ -1,4 +1,8 @@
use itertools::Itertools;
use std::num::ParseIntError;
use anyhow::Result;
use thiserror::Error;
use super::template::{DayTrait, ResultType};
@ -10,11 +14,82 @@ impl DayTrait for Day {
DAY_NUMBER
}
fn part1(&self, lines: String) -> Result<ResultType> {
Ok(ResultType::NoResult)
fn part1(&self, lines: &str) -> Result<ResultType> {
let vector = Day::parse(lines)?;
let max = vector.iter().max().ok_or(CalorieError::Empty)?;
Ok(ResultType::IntResult(*max))
}
fn part2(&self, lines: String) -> Result<ResultType> {
Ok(ResultType::NoResult)
fn part2(&self, lines: &str) -> Result<ResultType> {
let vector = Day::parse(lines)?;
let sum = vector.iter().sorted_by(|a, b| Ord::cmp(b, a)).take(3).sum();
Ok(ResultType::IntResult(sum))
}
}
impl Day {
fn parse(lines: &str) -> Result<Vec<i64>, CalorieError> {
Ok(lines
.split("\n")
.batching(|it| {
let result = it
.take_while(|line| line.len() != 0)
.map(|line| line.parse::<i64>())
.collect::<Result<Vec<_>, _>>()
.map(|lst: _| lst.iter().sum::<i64>());
result
.map(|value| if value == 0 { None } else { Some(value) })
.transpose()
})
.collect::<Result<_, _>>()?)
}
}
#[derive(Debug, Error)]
pub enum CalorieError {
#[error("Could not parse")]
ParseError(#[from] ParseIntError),
#[error("Did not get any values.")]
Empty,
}
#[cfg(test)]
mod test {
use super::*;
use crate::common::file::read_data;
use anyhow::Result;
#[test]
fn test_part1() -> Result<()> {
let day = Day {};
let lines = read_data(day.get_day_number(), "example01.txt")?;
let expected = ResultType::IntResult(24_000);
let result = day.part1(&lines)?;
assert_eq!(result, expected);
Ok(())
}
#[test]
fn test_part2() -> Result<()> {
let day = Day {};
let lines = read_data(day.get_day_number(), "example01.txt")?;
let expected = ResultType::IntResult(45_000);
let result = day.part2(&lines)?;
assert_eq!(result, expected);
Ok(())
}
#[test]
fn test_parse() -> Result<()> {
let lines = read_data(DAY_NUMBER, "example01.txt")?;
let expected = vec![6000, 4000, 11000, 24000, 10000];
let result = Day::parse(&lines)?;
assert_eq!(result, expected);
Ok(())
}
}

View file

@ -1,20 +1,50 @@
use anyhow::Result;
use super::template::{Day, ResultType};
use super::template::{DayTrait, ResultType};
pub struct Day;
const DAY_NUMBER: usize = 0;
impl DayTemplate for Day {
pub struct Day;
impl DayTrait for Day {
fn get_day_number(&self) -> usize {
DAY_NUMBER
}
fn part1(&self, lines: String) -> Result<ResultType> {
fn part1(&self, _lines: &str) -> Result<ResultType> {
Ok(ResultType::NoResult)
}
fn part2(&self, lines: String) -> Result<ResultType> {
fn part2(&self, _lines: &str) -> Result<ResultType> {
Ok(ResultType::NoResult)
}
}
#[cfg(test)]
mod test {
use super::*;
use crate::common::file::read_data;
use anyhow::Result;
#[test]
fn test_part1() -> Result<()> {
let day = Day {};
let lines = read_data(day.get_day_number(), "example01.txt")?;
let expected = ResultType::NoResult;
let result = day.part1(&lines)?;
assert_eq!(result, expected);
Ok(())
}
#[test]
fn test_part2() -> Result<()> {
let day = Day {};
let lines = read_data(day.get_day_number(), "example01.txt")?;
let expected = ResultType::NoResult;
let result = day.part2(&lines)?;
assert_eq!(result, expected);
Ok(())
}
}

21
src/days/day_provider.rs Normal file
View file

@ -0,0 +1,21 @@
use super::{day01, DayTrait};
use thiserror::Error;
const MAX_DAY: usize = 1;
pub fn get_day(day_num: usize) -> Result<Box<dyn DayTrait>, ProviderError> {
match day_num {
1 => Ok(Box::new(day01::Day)),
_ => Err(ProviderError::InvalidNumber(day_num)),
}
}
pub fn get_all_days() -> impl Iterator<Item = Box<dyn DayTrait>> {
(1..=MAX_DAY).map(|day_num| get_day(day_num).expect("Must never happen"))
}
#[derive(Debug, Error)]
pub enum ProviderError {
#[error("Not a valid day number: {0}")]
InvalidNumber(usize),
}

View file

@ -1,38 +1,6 @@
use anyhow::Result;
use self::{day01::Day, template::DayTrait};
use thiserror::Error;
pub use template::ResultType;
mod day01;
pub mod day_provider;
mod template;
#[derive(Debug, Error)]
pub enum TemplateError {
#[error("Not a valid day number: {0}")]
InvalidNumber(usize),
}
pub struct DayProvider {
days: Vec<Box<dyn DayTrait>>,
}
impl DayProvider {
pub fn create() -> DayProvider {
DayProvider {
days: vec![Box::new(Day)],
}
}
pub fn get_day(&self, day_num: usize) -> Result<&Box<dyn DayTrait>> {
Ok(self
.days
.get(day_num - 1)
.ok_or(TemplateError::InvalidNumber(day_num))?)
}
pub fn get_all_days(&self) -> &[Box<dyn DayTrait>] {
&self.days
}
}
pub use template::DayTrait;
pub use template::ResultType;

View file

@ -1,14 +1,16 @@
use anyhow::Result;
#[allow(dead_code)]
#[derive(Debug, PartialEq, Eq)]
pub enum ResultType {
IntResult(i64),
StringResult(String),
LinesResult(String),
LinesResult(Vec<String>),
NoResult,
}
pub trait DayTrait {
fn get_day_number(&self) -> usize;
fn part1(&self, lines: String) -> Result<ResultType>;
fn part2(&self, lines: String) -> Result<ResultType>;
fn part1(&self, lines: &str) -> Result<ResultType>;
fn part2(&self, lines: &str) -> Result<ResultType>;
}