summaryrefslogtreecommitdiff
path: root/2021/rust/day04
diff options
context:
space:
mode:
Diffstat (limited to '2021/rust/day04')
-rw-r--r--2021/rust/day04/Cargo.toml9
-rw-r--r--2021/rust/day04/src/main.rs217
2 files changed, 226 insertions, 0 deletions
diff --git a/2021/rust/day04/Cargo.toml b/2021/rust/day04/Cargo.toml
new file mode 100644
index 0000000..5f11cc4
--- /dev/null
+++ b/2021/rust/day04/Cargo.toml
@@ -0,0 +1,9 @@
+[package]
+name = "day04"
+version = "0.1.0"
+authors = ["cos <cos>"]
+edition = "2018"
+
+[dependencies]
+aoc = { path = "../../../common/rust/aoc" }
+anyhow = "1.0"
diff --git a/2021/rust/day04/src/main.rs b/2021/rust/day04/src/main.rs
new file mode 100644
index 0000000..e0457de
--- /dev/null
+++ b/2021/rust/day04/src/main.rs
@@ -0,0 +1,217 @@
+use {
+ anyhow::{
+ anyhow,
+ Context,
+ Result,
+ },
+ std::{
+ env::args,
+ fmt,
+ fmt::Display,
+ fs::File,
+ io::{
+ BufRead,
+ BufReader,
+ },
+ path::Path,
+ },
+};
+
+#[derive(Clone)]
+struct BingoBoard {
+ size: usize,
+ values: Vec<u8>,
+ called: Vec<bool>,
+ completed: Option<u8>,
+}
+
+impl BingoBoard {
+ fn new<'a, I: IntoIterator<Item = &'a u8> + Copy>(grid: I) -> Self {
+ let size = 5;
+ let values = (&grid).into_iter().copied().collect();
+ let called = vec![false; grid.into_iter().count()];
+ let completed = None;
+ Self {
+ size,
+ values,
+ called,
+ completed,
+ }
+ }
+
+ fn play(&mut self, num: &u8) -> Result<bool> {
+ if self.completed.is_some() {
+ return Ok(false);
+ }
+
+ for y in 0..self.size {
+ for x in 0..self.size {
+ let value = self.values.get(y * self.size + x)
+ .ok_or(anyhow!("Invalid BingoBoard"))?;
+
+ if value == num {
+ self.called[y * self.size + x] = true;
+ }
+ }
+ }
+ for y in 0..self.size {
+ let mut count = 0;
+ for x in 0..self.size {
+ if *self.called.get(y * self.size + x).ok_or(anyhow!("Invalid BingoBoard"))? {
+ count += 1;
+ }
+ if count == self.size {
+ self.completed = Some(*num);
+ return Ok(true);
+ }
+ }
+ }
+ for x in 0..self.size {
+ let mut count = 0;
+ for y in 0..self.size {
+ if *self.called.get(y * self.size + x).ok_or(anyhow!("Invalid BingoBoard"))? {
+ count += 1;
+ }
+ if count == self.size {
+ self.completed = Some(*num);
+ return Ok(true);
+ }
+ }
+ }
+ Ok(false)
+ }
+
+ fn sum_unmarked(&self) -> Result<usize> {
+ let mut sum = 0;
+ for y in 0..self.size {
+ for x in 0..self.size {
+ let value = self.values.get(y * self.size + x)
+ .ok_or(anyhow!("Invalid BingoBoard"))?;
+
+ if ! *self.called.get(y * self.size + x).ok_or(anyhow!("Invalid BingoBoard"))? {
+ sum += *value as usize;
+ }
+ }
+ }
+ Ok(sum)
+ }
+}
+
+impl Display for BingoBoard {
+ fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
+ let mut s = String::new();
+ for y in 0..self.size {
+ for x in 0..self.size {
+ let value = self.values.get(y * self.size + x).ok_or(std::fmt::Error)?;
+ if *self.called.get(y * self.size + x).ok_or(std::fmt::Error)? {
+ s += &format!("[{:2}] ", value);
+ } else {
+ s += &format!(" {:2} ", value);
+ }
+ }
+ s += "\n";
+ }
+ if let Some(won) = self.completed {
+ s += &format!("Winning number: {}\n", won);
+ }
+ write!(f, "{}", s)
+ }
+}
+
+#[derive(Clone)]
+struct BingoGame {
+ numbers: Vec<u8>,
+ boards: Vec<BingoBoard>,
+}
+
+impl BingoGame {
+ fn new() -> Self {
+ Self {
+ numbers: vec![],
+ boards: vec![],
+ }
+ }
+}
+
+fn read_input<T: AsRef<Path>>(filename: T) -> Result<BingoGame> {
+ let reader = BufReader::new(File::open(filename)?);
+ let mut game = BingoGame::new();
+ let mut boards: Vec<BingoBoard> = vec![];
+
+ let mut lines = reader.lines();
+ let draw_line = lines.next().ok_or(anyhow!("Could not parse draw_line"))?
+ .map_err(|err| anyhow!("{}", err))?;
+ let mut buffer: Vec<u8> = vec![];
+ for line in lines {
+ let l = line?;
+ if l.is_empty() {
+ if ! buffer.is_empty() {
+ let board = BingoBoard::new(&buffer);
+ boards.push(board);
+ }
+ buffer = vec![];
+ } else {
+ buffer.append(&mut l.split_whitespace().map(|v| v.parse()
+ .map_err(|_| anyhow!("Parse error"))).collect::<Result<_>>()?);
+ }
+ }
+ let board = BingoBoard::new(&buffer);
+ boards.push(board);
+
+ game.numbers = draw_line.split(',').map(|v| v.parse().map_err(|err| anyhow!("{}", err)))
+ .collect::<Result<_>>()?;
+ game.boards = boards;
+
+ Ok(game)
+}
+
+fn part1(game: &BingoGame) -> Result<usize> {
+ let mut game_mut = game.clone();
+ for num in &game_mut.numbers {
+ for board in &mut game_mut.boards {
+ if board.play(num)? {
+ // println!("{}", board);
+ return Ok(board.sum_unmarked()? * *num as usize);
+ }
+ }
+ }
+ Err(anyhow!("No winner"))
+}
+
+fn part2(game: &BingoGame) -> Result<usize> {
+ let mut game_mut = game.clone();
+ let mut last_winner = None;
+ for num in &game_mut.numbers {
+ for (index, board) in &mut game_mut.boards.iter_mut().enumerate() {
+ if board.play(num)? {
+ last_winner = Some(index);
+ }
+ }
+ }
+ if let Some(winner) = last_winner {
+ let board = &game_mut.boards[winner];
+ let unmarked_sum = board.sum_unmarked()?;
+ let won = board.completed.ok_or(anyhow!("No winning number"))? as usize;
+ // println!("Winning board: {} ({} * {})", winner + 1, unmarked_sum, won);
+ // println!("{}", board);
+ Ok(unmarked_sum * won)
+ } else {
+ Err(anyhow!("No winner"))
+ }
+}
+
+fn main() -> Result<()> {
+ let ( do_part_1, do_part_2 ) = aoc::do_parts();
+
+ let filename = args().nth(1).ok_or(anyhow!("Missing input filename"))?;
+ let input = read_input(filename).context("Could not read input")?;
+ if do_part_1 {
+ let solution = part1(&input).context("No solution for part 1")?;
+ println!("Part1, solution found to be: {}", solution);
+ }
+ if do_part_2 {
+ let solution = part2(&input).context("No solution for part 2")?;
+ println!("Part2, solution found to be: {}", solution);
+ }
+ Ok(())
+}