diff options
Diffstat (limited to '2020/rust/day07')
-rw-r--r-- | 2020/rust/day07/Cargo.toml | 13 | ||||
-rw-r--r-- | 2020/rust/day07/src/main.rs | 172 |
2 files changed, 185 insertions, 0 deletions
diff --git a/2020/rust/day07/Cargo.toml b/2020/rust/day07/Cargo.toml new file mode 100644 index 0000000..c30cde6 --- /dev/null +++ b/2020/rust/day07/Cargo.toml @@ -0,0 +1,13 @@ +[package] +name = "day07" +version = "0.1.0" +authors = ["cos <cos>"] +edition = "2018" + +[dependencies] +aoc = { path = "../aoc" } +anyhow = "1.0" +daggy = "0.7.0" +#petgraph = "0.4.13" +petgraph = "0.5.1" +regex = "1.4" diff --git a/2020/rust/day07/src/main.rs b/2020/rust/day07/src/main.rs new file mode 100644 index 0000000..d643241 --- /dev/null +++ b/2020/rust/day07/src/main.rs @@ -0,0 +1,172 @@ +use anyhow::{anyhow, Result}; +use daggy::{Dag, Walker, NodeIndex}; +use petgraph::dot::{Dot, Config}; +use regex::Regex; +use std::collections::{HashMap, HashSet}; +use std::env::args; +use std::fs::File; +use std::io::{BufRead, BufReader, BufWriter}; +use std::io::prelude::*; +use std::path::Path; + +type BagCount = usize; +type BagColor = String; +type Rules = Dag<BagColor, BagCount>; +type Index = HashMap<String, NodeIndex>; + +// The example data in +// simplified form: A | E | D +// 1 x B | 1 x G | / \ +// A contain 1B, 2C. 2 x C | 2 x H | | A | +// D contain 3B, 4C. D | G | |/ \| +// B contain 1E. 3 x B | 3 x F | B __C +// C contain 2E, 9F. 4 x C | 4 x I | |/ | +// E contain 1G, 2H. B | H | E | +// G contain 3F, 4I. 1 x E | 5 x F | /| / +// H contain 5F, 6I. C | 6 x I | H G / +// F contain NONE. 2 x E | F | / \|/ +// I contain NONE. 9 x F | I | I F + +fn read_input<T: AsRef<Path>>(filename: T) -> Result<(Rules, Index)> { + let f = File::open(filename)?; + let reader = BufReader::new(f); + + let outer_re = Regex::new( + r"(?x) + ^(?P<outer>[a-z\s]+)\ bags?\ contain\ + (?P<inner>(?:(?:\d+\ [a-z\s]+|no\ other)\ bags?[.,]\s?)+)$" + )?; + + let inner_re = Regex::new( + r"(?x) + (?:(?:(?P<count>\d+)\ (?P<color>[a-z\s]+)|(?P<empty>no\ other))\ bags?[.,]\s?)" + )?; + + let mut rules: Rules = Dag::new(); + let mut rules_index = HashMap::new(); + + for line in reader.lines() { + let line = line?; + let caps_outer = outer_re.captures(&line).ok_or_else(|| anyhow!("Invalid input"))?; + let container_color = caps_outer.name("outer") + .ok_or_else(|| anyhow!("Invalid input"))?.as_str().to_string(); + let parent_index = match rules_index.get(&String::from(&container_color)) { + None => { + rules.add_node(container_color.clone()) + }, + Some(existing_index) => { + *existing_index + }, + }; + rules_index.insert(container_color, parent_index); + for m in inner_re.find_iter(caps_outer.name("inner") + .ok_or_else(|| anyhow!("Invalid input"))?.as_str()) + { + let s = m.as_str(); + let caps_inner = inner_re.captures(&s).ok_or_else(|| anyhow!("Invalid input"))?; + match caps_inner.name("empty") { + Some(_) => { }, + None => { + let content_color = caps_inner.name("color") + .ok_or_else(|| anyhow!("Invalid input"))?.as_str(); + let content_count: usize = caps_inner.name("count") + .ok_or_else(|| anyhow!("Invalid input"))?.as_str().parse()?; + match rules_index.get(content_color) { + None => { + let (_, child_index) = rules.add_child( + parent_index, content_count, String::from(content_color) + ); + rules_index.insert(String::from(content_color), child_index); + }, + Some(existing_index) => { + let _ = rules.add_edge(parent_index, *existing_index, content_count); + }, + } + } + } + } + } + + Ok((rules, rules_index)) +} + +fn plot_graph(rules: Rules) -> Result<()>{ + let f = File::create("graph.dot")?; + let mut writer = BufWriter::new(f); + let g = rules.graph(); + write!(writer, "{}", Dot::with_config(g, &[Config::EdgeNoLabel]))?; + Ok(()) +} + +fn recurse_part1(start: NodeIndex, rules: &Rules, mut visited: &mut HashSet<BagColor>) -> usize { + let mut ans = 0; + let node = rules.node_weight(start).unwrap(); + + if ! visited.contains(node) { + ans += 1; + visited.insert(String::from(node)); + let mut parent_walker = rules.parents(start); + while let Some(walk) = parent_walker.walk_next(&rules) { + let ( _edge, next_node_index ) = walk; + ans += recurse_part1(next_node_index, rules, &mut visited); + } + } + + ans +} + +fn recurse_part2(start: NodeIndex, rules: &Rules) -> usize { + let mut ans=0; + + rules.node_weight(start).unwrap(); + let mut child_walker = rules.children(start); + while let Some(walk) = child_walker.walk_next(&rules) { + let ( edge_index, next_node_index ) = walk; + let content_count = rules.edge_weight(edge_index).unwrap(); + ans += content_count * (1 + recurse_part2(next_node_index, rules)); + } + + ans +} + +fn part1(rules: &Rules, index: &Index) -> Result<usize> { + let start_index = index.get(&String::from("shiny gold")).ok_or(anyhow!("Bag is missing"))?; + let mut visited: HashSet<BagColor> = HashSet::new(); + let ans = recurse_part1(*start_index, rules, &mut visited); + + Ok(ans-1) +} + +fn part2(rules: &Rules, index: &Index) -> Result<usize> { + let start_index = index.get(&String::from("shiny gold")).ok_or(anyhow!("Bag is missing"))?; + Ok(recurse_part2(*start_index, rules)) +} + +fn main() { + let ( do_part_1, do_part_2 ) = aoc::do_parts(); + + let filename = match args().nth(1) { + Some(f) => f, + None => { + eprintln!("Missing input filename"); + std::process::exit(1); + }, + }; + match read_input(filename) { + Ok((input, index)) => { + if do_part_1 { + let solution = part1(&input, &index); + println!("Part1, {:?}", solution); + } + if do_part_2 { + let solution = part2(&input, &index); + println!("Part2, {:?}", solution); + } + match plot_graph(input) { + Ok(_) => println!("Wrote a graphviz file. Please feel free to plot it."), + Err(err) => eprintln!("Could not write graphviz file: {}", err), + } + }, + Err(err) => eprintln!("Could not read input: {}", err), + } +} |