use std::{env, fs}; use std::collections::{HashMap, HashSet, VecDeque}; use Module::{Broadcaster, Conjunction, FlipFlop, Counter}; use crate::PulseKind::{High, Low}; #[derive(Copy, Clone, Debug, Eq, PartialEq, Hash)] enum PulseKind { High, Low } #[derive(Clone, Debug)] struct Pulse { kind: PulseKind, origin: String, destination: String } enum Module { Broadcaster { out: Vec }, Conjunction { out: Vec, memory: HashMap }, FlipFlop { out: Vec, state: bool }, Counter { out: Vec, high: i32, low: i32} } fn main() { println!("Hello, AoC day 19!"); let args: Vec = env::args().collect(); if args.len() != 2 { println!("wrong number of arguments!"); std::process::exit(1); } let file_path = &args[1]; let contents = fs::read_to_string(file_path).expect("Should have been able to read the file"); let mut rack: HashMap = HashMap::new(); for line in contents.lines() { let (name, connections) = line.split_once(" -> ").unwrap(); let outputs: Vec = connections.split(", ").map(|it| it.to_string()).collect(); match name.chars().nth(0).unwrap() { '&' => rack.insert(name[1..].to_string(), Conjunction { out: outputs, memory: HashMap::new() } ), '%' => rack.insert(name[1..].to_string(), FlipFlop { out: outputs, state: false } ), _ => rack.insert(name.to_string(), Broadcaster { out: outputs } ), }; } // print out the rack and also tally up the listed outputs println!("the rack looks like this:"); let mut all_listed_outputs: HashSet = HashSet::new(); for (key, value) in &rack { match value { Broadcaster { out } => { println!("{} -> {}", key, out.join(", ")); for o in out { all_listed_outputs.insert(o.to_string()); } } Conjunction { out, .. } => { println!("&{} -> {}", key, out.join(", ")); for o in out { all_listed_outputs.insert(o.to_string()); } } FlipFlop { out, .. } => { println!("%{} -> {}", key, out.join(", ")); for o in out { all_listed_outputs.insert(o.to_string()); } } Counter { out, .. } => { println!("${} -> {}", key, out.join(", ")); for o in out { all_listed_outputs.insert(o.to_string()); } } } } // make sure all the outputs are present as outputs for label in all_listed_outputs { if rack.get(&label).is_none() { rack.insert(label, Counter{ out: Vec::new(), high: 0, low: 0}); } } println!("let's fix up those conjunction modules so they know about all their inputs and are initialized properly to low."); let mut conjunctions_and_inputs: HashMap> = HashMap::new(); // go through and take note of every cnojunction module in our conjunctions_and_inputs collection for (label, _module) in &rack { match rack.get(label).unwrap() { Conjunction{ out:_,memory:_} => { conjunctions_and_inputs.insert(label.to_string(), Vec::new()); } _ => {} // get fucked!!! }; } // look at every module. if it's got a conjunction in the output, list it in the collection of conjunctions and inputs for (label, value) in &rack { match value { Broadcaster{ out } => { for o in out { if let Conjunction { .. } = rack.get(o).unwrap() { conjunctions_and_inputs.get_mut(o).unwrap().push(label.to_string()) } } } Conjunction{ out, .. } => { for o in out { if let Conjunction { .. } = rack.get(o).unwrap() { conjunctions_and_inputs.get_mut(o).unwrap().push(label.to_string()) } } } FlipFlop{ out, .. } => { for o in out { if let Conjunction { .. } = rack.get(o).unwrap() { conjunctions_and_inputs.get_mut(o).unwrap().push(label.to_string()) } } } Counter { out, .. } => { for o in out { if let Conjunction { .. } = rack.get(o).unwrap() { conjunctions_and_inputs.get_mut(o).unwrap().push(label.to_string()) } } } } } // connect each input t othe cnojunctino module for (destination, inputs) in conjunctions_and_inputs { for start in inputs { connect(start, destination.to_string(), &mut rack); } } let mut high = 0; let mut low = 0; for _i in 0..1000 { let (h, l) = push_the_button_and_count(&mut rack); println!("there were {high} high pulses and {low} low pulses."); high += h; low += l; } println!("after 1000 pushes, we have sent {low} low pulses and {high} high pulses"); let product = (high as i64) * (low as i64); println!("that's aproduct of {product}"); println!("the set of things outputting to rx is:"); let mut goes_to_rx: HashSet = HashSet::new(); for (name, module) in &rack { match module { Broadcaster { out, .. } => { if out.contains(&String::from("rx")) {goes_to_rx.insert(name.to_string());} } Conjunction { out, .. } => {if out.contains(&String::from("rx")) {goes_to_rx.insert(name.to_string());}} FlipFlop { out, .. } => {if out.contains(&String::from("rx")) {goes_to_rx.insert(name.to_string());}} Counter { out, .. } => {if out.contains(&String::from("rx")) {goes_to_rx.insert(name.to_string());}} } } println!("{goes_to_rx:?}"); if let Conjunction{memory, ..} = rack.get(goes_to_rx.iter().nth(0).unwrap()).unwrap() { for x in memory.keys() { print!("{x}, "); } println!("are all options for what we're interestd in"); } let things_to_watch: HashSet = String::from("qq,sj,ls,bg").split(',').map(|x| x.to_string()).collect(); let mut i=1001; loop { let interest = push_the_button_and_watch(&mut rack, &things_to_watch, goes_to_rx.iter().nth(0).unwrap().to_string()); if interest { println!("interesting things have happened on press {i}"); } if i % 100_000 == 0 { if let Counter { low, .. } = rack.get("rx").unwrap() { println!("rx has been hit with a low pulse {low} times after {i} buttons."); } } i+=1 } } fn connect(origin: String, dest_conj: String, rack: &mut HashMap) { let conjunction_module = rack.get_mut(&dest_conj).unwrap(); if let Conjunction { memory, ..} = conjunction_module { memory.insert(origin, Low); } } fn push_the_button_and_count(rack: &mut HashMap) -> (i32, i32) { let mut pulses = VecDeque::from([Pulse { kind: Low, origin: String::from("button"), destination: String::from("broadcaster") }]); let mut high = 0; let mut low = 0; while !pulses.is_empty() { let pulse = pulses.pop_front().unwrap(); let _origin = pulse.origin.to_string(); let _dest = pulse.destination.to_string(); let _kind = match pulse.kind { High => { "high" } Low => { "low" } }; //println!("{origin} -{kind}-> {dest}"); match pulse.kind { High => { high += 1; } Low => { low += 1; } } let new_pulses = handle_pulse(&pulse, rack); for x in new_pulses { pulses.push_back(x); } } return (high, low); } fn push_the_button_and_watch(rack: &mut HashMap, modules_to_watch: &HashSet, conjunction: String) -> bool { let mut pulses = VecDeque::from([Pulse { kind: Low, origin: String::from("button"), destination: String::from("broadcaster") }]); let mut high = 0; let mut low = 0; let mut of_interest = false; while !pulses.is_empty() { let pulse = pulses.pop_front().unwrap(); let _origin = pulse.origin.to_string(); let _dest = pulse.destination.to_string(); let _kind = match pulse.kind { High => { "high" } Low => { "low" } }; if modules_to_watch.contains(&_origin) && pulse.kind == Low { if let Conjunction {memory, ..} = &rack.get(&conjunction).unwrap() { let pulses: HashSet<&PulseKind>= modules_to_watch.iter().map(|x| memory.get(x).unwrap()).collect(); if pulses.contains(&High) { of_interest = true; } } } //println!("{origin} -{kind}-> {dest}"); let new_pulses = handle_pulse(&pulse, rack); for x in new_pulses { pulses.push_back(x); } } return of_interest; } fn handle_pulse(pulse: &Pulse, rack: &mut HashMap) -> Vec { // a pulse has arrived!~ let's handle it. let destination_module = rack.get_mut(&pulse.destination).unwrap(); match destination_module { Broadcaster{ out } => { // broadcast to all the outputs! return out.iter().map(|d| Pulse { kind: pulse.kind, origin: pulse.destination.to_string(), destination: d.to_string() }).collect(); } Conjunction{ out, memory} => { // update the memory. memory.insert(pulse.origin.to_string(), pulse.kind); // go through the memories and look for one thats low, in which case we output a low pulse. for value in memory.values() { if *value == Low { // if any of the memories is low, output a high pulse return out.iter().map(|d| Pulse { kind: High, origin: pulse.destination.to_string(), destination: d.to_string() }).collect(); } } // if we've gotten here, all the memories were high, so send a low pulse out.iter().map(|d| Pulse { kind: Low, origin: pulse.destination.to_string(), destination: d.to_string() }).collect()} FlipFlop{ out, state} => { match pulse.kind { High => { // on a high pulse, flip flops produce nothijng Vec::new() }, Low => { // on a low pulse, we flip it, and then send an appropriate pulse (high if it's on, low if it's off) *state = !*state; out.iter().map(|d| Pulse { kind: match state { true => High, false => Low }, origin: pulse.destination.to_string(), destination: d.to_string() }).collect() } } } Counter { high, low, ..} => { match pulse.kind { High => { *high += 1; }, Low => { *low += 1; } } Vec::new() } } }