summary refs log tree commit diff homepage
path: root/2016/src/bin/day04.rs
blob: 8e724c220d25dcfa397843e7a9767250e929d83b (plain) (blame)
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
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
use std::io::{self, Read};
use std::str::FromStr;

struct Room {
    name: String,
    sector_id: u32,
    checksum: String,
}

impl FromStr for Room {
    type Err = ();
    fn from_str(s: &str) -> Result<Self, ()> {
        let mut iter = s.trim_right_matches(']')
            .rsplitn(3, |c| c == '-' || c == '[');

        let checksum = iter.next().ok_or(())?;
        let sector_id = iter.next().ok_or(())?;
        let name = iter.next().ok_or(())?;

        Ok(Room {
            name: name.into(),
            sector_id: sector_id.parse().map_err(|_| ())?,
            checksum: checksum.into(),
        })
    }
}

impl Room {
    fn checksum(&self) -> String {
        let mut letters = [
            (0, 'a'), (0, 'b'), (0, 'c'), (0, 'd'), (0, 'e'), (0, 'f'), (0, 'g'), (0, 'h'),
            (0, 'i'), (0, 'j'), (0, 'k'), (0, 'l'), (0, 'm'), (0, 'n'), (0, 'o'), (0, 'p'),
            (0, 'q'), (0, 'r'), (0, 's'), (0, 't'), (0, 'u'), (0, 'v'), (0, 'w'), (0, 'x'),
            (0, 'y'), (0, 'z'),
        ];

        for letter in self.name.chars().filter(|c| c.is_alphabetic()) {
            let index = letter as u8 - b'a';
            letters[index as usize].0 -= 1;
        }

        letters.sort();

        letters.into_iter()
            .map(|l| l.1)
            .take(5)
            .collect()
    }

    fn verify_checksum(&self) -> bool {
        self.checksum == self.checksum()
    }

    fn decrypt_name(&self) -> String {
        self.name.chars()
            .map(|c| {
                match c {
                    '-' => ' ',
                    _ => rotate(c, self.sector_id),
                }
            })
            .collect()
    }
}

fn rotate(c: char, n: u32) -> char {
    let c = c as u8 + (n % 26) as u8;
    if c > b'z' {
        (c - 26) as char
    } else {
        c as char
    }
}

fn solve1(input: &str) -> u32 {
    input.lines()
        .map(str::parse)
        .map(Result::unwrap)
        .filter(Room::verify_checksum)
        .map(|room| room.sector_id)
        .sum()
}

fn solve2(input: &str) -> Option<u32> {
    input.lines()
        .map(str::parse)
        .map(Result::unwrap)
        .filter(Room::verify_checksum)
        .filter_map(|r| {
            match r.decrypt_name().as_str() {
                "northpole object storage" => Some(r.sector_id),
                _ => None,
            }
        })
        .next()
}

fn main() {
    let mut input = String::new();
    io::stdin().read_to_string(&mut input).unwrap();

    println!("Part 1: {}", solve1(&input));
    if let Some(sector_id) = solve2(&input) {
        println!("Part 2: {}", sector_id);
    }
}

#[test]
fn part1() {
    let input = "
aaaaa-bbb-z-y-x-123[abxyz]
a-b-c-d-e-f-g-h-987[abcde]
not-a-real-room-404[oarel]
totally-real-room-200[decoy]
";
    assert_eq!(1514, solve1(&input[1..]));
}

#[test]
fn part2() {
    let room: Room = "qzmt-zixmtkozy-ivhz-343[zimth]".parse().unwrap();
    assert_eq!("very encrypted name", room.decrypt_name());
}
ng uiFmt format stringsJune McEnroe 2018-08-07Handle PART and QUIT without messagesJune McEnroe 2018-08-07Make safe filling the who bufferJune McEnroe 2018-08-07Add reverse and reset IRC formatting codesJune McEnroe 2018-08-06Rewrite line editing again, add formattingJune McEnroe 2018-08-06Fix allocation size in vaswprintfJune McEnroe This is so embarrassing. It only started crashing once it had strings that were long enough, and then it took me so long to notice this mistake. I was worried I was still doing va_list wrong somehow. 2018-08-06Implement word wrappingJune McEnroe 2018-08-06Use wchar_t strings for all of UIJune McEnroe vaswprintf is a nightmare. 2018-08-06Rename line editing functionsJune McEnroe 2018-08-05Initialize all possible color pairsJune McEnroe This is actually possible with use_default_colors! 2018-08-05Refactor color initializationJune McEnroe 2018-08-05Add ^L redrawJune McEnroe 2018-08-05Use 16 colors if availableJune McEnroe Fall back to using bold if there are only 8 colors. This also allowed bright background colors in 16-color terminals. I must port this system to torus. I'll be able to remove the awful termcap patch hack. 2018-08-05Limit parsed colors to number of mIRC colorsJune McEnroe Oh boy that's embarrassing. 2018-08-04Show source link on exitJune McEnroe 2018-08-04Implement line editing, scrollingJune McEnroe Don't really have a way to implement the M-* keys, and currently missing C-w. 2018-08-04Handle /topicJune McEnroe