-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path8.py
62 lines (43 loc) · 1.54 KB
/
8.py
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
from typing import Callable
from math import lcm
import re
from read_input import read_input
DAY = 8
def part1() -> int:
instructions, _, *lines = read_input(DAY)
instructions = instructions.strip()
maps = parse_maps(lines)
return iterate_maps('AAA', check_end_part1, maps, instructions)
def parse_maps(lines: list[str]) -> dict[str, tuple[str, str]]:
maps = {}
line_regex = re.compile("(\w{3}) = \((\w{3}), (\w{3})\)")
for line in lines:
name, left, right = list(line_regex.search(line).groups())
maps[name] = (left, right)
return maps
def check_end_part1(current: str) -> bool:
return current == 'ZZZ'
def iterate_maps(start: str, end_check: Callable[[str], bool], maps: dict[str, tuple[str, str]], instructions) -> int:
map_name = start
count = 0
while True:
for direction in instructions:
map_next = maps[map_name]
map_name = map_next[0] if direction == 'L' else map_next[1]
count += 1
if end_check(map_name):
return count
def part2() -> int:
instructions, _, *lines = read_input(8)
instructions = instructions.strip()
maps = parse_maps(lines)
current_maps = [name for name in maps.keys() if name[-1] == 'A']
counts = []
for map_cur in current_maps:
counts.append(iterate_maps(map_cur, check_end_part2, maps, instructions))
return lcm(*counts)
def check_end_part2(current: str) -> bool:
return current[-1] == 'Z'
if __name__ == "__main__":
print(part1())
print(part2())