summaryrefslogtreecommitdiff
path: root/src/bin/day7.rs
blob: 60b08b103a02e2ea69d71cf2549aa896b311c932 (plain)
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
static DAY: u8 = 7;

fn main() {
    let input = advent::read_lines(DAY);
    println!("{DAY}a: {}", calibration_result(&input));
    println!("{DAY}b: {}", 0);
}

fn calculateable(result: i64, operands: &[i64]) -> bool {
    if operands.len() == 1 {
        return result == operands[0];
    }
    let rem = &operands[..=operands.len()-2];
    let op = operands[operands.len()-1];
    calculateable(result - op, rem) || (result % op == 0 && calculateable(result / op, rem))
}

fn calibration_result(input: &[String]) -> i64 {
    let mut sum = 0;
    for line in input {
        let (result, operands) = line.split_once(": ").unwrap();
        let result = result.parse::<i64>().unwrap();
        let operands = operands.split(" ").map(|x| x.parse::<i64>().unwrap()).collect::<Vec<_>>();
        if calculateable(result, &operands) {
            sum += result;
        }
    }
    sum
}

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

    #[test]
    fn test() {
        let input = [
            "190: 10 19",
            "3267: 81 40 27",
            "83: 17 5",
            "156: 15 6",
            "7290: 6 8 6 15",
            "161011: 16 10 13",
            "192: 17 8 14",
            "21037: 9 7 18 13",
            "292: 11 6 16 20",
        ].iter().map(|&x| String::from(x)).collect::<Vec<_>>();
        assert_eq!(calibration_result(&input), 3749);
    }
}