use std::collections::HashMap; use std::error::Error; use std::io::{self, Read}; use std::str::FromStr; use regex::Regex; fn main() -> Result<(), Box> { let mut input = String::new(); io::stdin().read_to_string(&mut input)?; let output = solve(&input)?; println!("{}", output); Ok(()) } fn solve(input: &str) -> Result> { let claims: Vec<_> = input .lines() .map(str::trim) .map(Claim::from_str) .collect::>()?; let fabric = claims.iter().fold( HashMap::new(), |mut fabric: HashMap<(usize, usize), Vec>, claim| { for square_inch in claim.square_inches() { fabric .entry(square_inch) .and_modify(|ids| ids.push(claim.id)) .or_insert_with(|| vec![claim.id]); } fabric }, ); let values: Vec<_> = fabric.values().collect(); let output = values .iter() .map(|ids| ids[0]) .find(|id| !values.iter().any(|ids| ids.len() > 1 && ids.contains(id))) .unwrap() .to_string(); Ok(output) } #[test] fn test_solve() { let input = r" #1 @ 1,3: 4x4 #2 @ 3,1: 4x4 #3 @ 5,5: 2x2 " .trim(); assert_eq!(&solve(input).unwrap(), "3"); } #[derive(Debug, PartialEq, Eq)] struct Claim { id: usize, x: usize, y: usize, width: usize, height: usize, } impl Claim { fn square_inches(&self) -> impl Iterator + '_ { (self.x..self.x + self.width) .flat_map(move |x| (self.y..self.y + self.height).map(move |y| (x, y))) } } #[test] fn test_claim_square_inches() { let claim = Claim { id: 1, x: 2, y: 3, width: 4, height: 4, }; let square_inches: Vec<_> = claim.square_inches().collect(); assert_eq!(square_inches.len(), 16); assert!(square_inches.contains(&(2, 3))); assert!(square_inches.contains(&(5, 3))); assert!(square_inches.contains(&(5, 6))); assert!(square_inches.contains(&(2, 6))); } impl FromStr for Claim { type Err = Box; fn from_str(s: &str) -> Result { let re = Regex::new(r"^#(?P\d+) @ (?P\d+),(?P\d+): (?P\d+)x(?P\d+)$") .unwrap(); let captures = re.captures(s).unwrap(); let id = captures.name("id").unwrap().as_str().parse()?; let x = captures.name("x").unwrap().as_str().parse()?; let y = captures.name("y").unwrap().as_str().parse()?; let width = captures.name("width").unwrap().as_str().parse()?; let height = captures.name("height").unwrap().as_str().parse()?; Ok(Claim { id, x, y, width, height, }) } } #[test] fn test_claim_from_str() { let claim = Claim::from_str("#1 @ 1,3: 4x4").unwrap(); assert_eq!( claim, Claim { id: 1, x: 1, y: 3, width: 4, height: 4 } ); }