1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
//! # Day 2: I Was Told There Would Be No Math
//!
//! The elves are running low on wrapping paper, and so they need to submit an order for more. They
//! have a list of the dimensions (length `l`, width `w`, and height `h`) of each present, and only
//! want to order exactly as much as they need.
//!
//! Fortunately, every present is a box (a perfect [right rectangular prism]), which makes
//! calculating the required wrapping paper for each gift a little easier: find the surface area of
//! the box, which is `2*l*w + 2*w*h + 2*h*l`. The elves also need a little extra paper for each
//! present: the area of the smallest side.
//!
//! For example:
//!
//! - A present with dimensions `2x3x4` requires `2*6 + 2*12 + 2*8 = 52` square feet of wrapping
//!   paper plus `6` square feet of slack, for a total of `58` square feet.
//! - A present with dimensions `1x1x10` requires `2*1 + 2*10 + 2*10 = 42` square feet of wrapping
//!   paper plus `1` square foot of slack, for a total of `43` square feet.
//!
//! All numbers in the elves' list are in feet. How many total **square feet of wrapping paper**
//! should they order?
//!
//! [right rectangular prism]: https://en.wikipedia.org/wiki/Cuboid#Rectangular_cuboid
//!
//! ## Part Two
//!
//! The elves are also running low on ribbon. Ribbon is all the same width, so they only have to
//! worry about the length they need to order, which they would again like to be exact.
//!
//! The ribbon required to wrap a present is the shortest distance around its sides, or the smallest
//! perimeter of any one face. Each present also requires a bow made out of ribbon as well; the feet
//! of ribbon required for the perfect bow is equal to the cubic feet of volume of the present.
//! Don't ask how they tie the bow, though; they'll never tell.
//!
//! For example:
//!
//! - A present with dimensions `2x3x4` requires `2+2+3+3 = 10` feet of ribbon to wrap the present
//!   plus `2*3*4 = 24` feet of ribbon for the bow, for a total of `34` feet.
//! - A present with dimensions `1x1x10` requires `1+1+1+1 = 4` feet of ribbon to wrap the present
//!   plus `1*1*10 = 10` feet of ribbon for the bow, for a total of `14` feet.
//!
//! How many total **feet of ribbon** should they order?

use anyhow::{anyhow, ensure, Result};

pub const INPUT: &str = include_str!("d02.txt");

pub fn solve_part_one(input: &str) -> Result<u32> {
    let boxes = parse_input(input)?;

    Ok(boxes
        .iter()
        .map(|b| {
            let lw = b.0 * b.1;
            let wh = b.1 * b.2;
            let hl = b.2 * b.0;
            let slack = lw.min(wh).min(hl);
            2 * (lw + wh + hl) + slack
        })
        .sum())
}

pub fn solve_part_two(input: &str) -> Result<u32> {
    let boxes = parse_input(input)?;

    Ok(boxes
        .iter()
        .map(|b| {
            // By subtracting the biggest value, we get the sum of the two smallest ones.
            let biggest = b.0.max(b.1).max(b.2);
            2 * (b.0 + b.1 + b.2 - biggest) + b.0 * b.1 * b.2
        })
        .sum())
}

fn parse_input(input: &str) -> Result<Vec<(u32, u32, u32)>> {
    input
        .lines()
        .map(|l| {
            let mut parts = l.split('x');
            let l = parts.next().ok_or_else(|| anyhow!("missing length"))?.parse()?;
            let w = parts.next().ok_or_else(|| anyhow!("missing width"))?.parse()?;
            let h = parts.next().ok_or_else(|| anyhow!("missing height"))?.parse()?;
            ensure!(parts.next().is_none(), "too many parts");
            Ok((l, w, h))
        })
        .collect()
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn part_one() {
        assert_eq!(58, solve_part_one("2x3x4").unwrap());
        assert_eq!(43, solve_part_one("1x1x10").unwrap());
    }

    #[test]
    fn part_two() {
        assert_eq!(34, solve_part_two("2x3x4").unwrap());
        assert_eq!(14, solve_part_two("1x1x10").unwrap());
    }
}