-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathday10.py
105 lines (76 loc) · 2.37 KB
/
day10.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
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
import argparse
import os
from collections import deque
CHUNK_MAP = {
"(": ")",
"<": ">",
"[": "]",
"{": "}",
}
POINTS = {")": 3, "]": 57, "}": 1197, ">": 25137}
POINTS2 = {")": 1, "]": 2, "}": 3, ">": 4}
def read_input(filepath: str):
with open(filepath, "r") as f:
return [l.strip() for l in f.readlines()]
def init_parser() -> str:
parser = argparse.ArgumentParser(description="Advent of Code day 10 solution.")
parser.add_argument(
"input", metavar="FILE", type=str, nargs=1, help="Path to input data."
)
args = parser.parse_args()
return os.path.realpath(args.input[0])
def validate_chunk(line: str, start: int) -> int:
if line[start] not in CHUNK_MAP.keys():
return 0
stack = deque()
stack.append(CHUNK_MAP[line[start]])
for i in range(start + 1, len(line)):
char = line[i]
if char in CHUNK_MAP.keys():
stack.append(CHUNK_MAP[char])
elif char in CHUNK_MAP.values() and len(stack) > 0:
match = stack.pop()
if char != match:
return POINTS[char]
return 0
def find_chunks(line: str) -> int:
for i in range(len(line)):
if (points := validate_chunk(line, i)) != 0:
return points
return 0
def part_1(lines: list[str]) -> int:
points = 0
for line in lines:
points += find_chunks(line)
return points
def complete_line(line: str) -> str:
stack = deque()
for i in range(len(line)):
char = line[i]
if char in CHUNK_MAP.keys():
stack.append(CHUNK_MAP[char])
elif char in CHUNK_MAP.values() and len(stack) > 0:
stack.pop()
stack.reverse()
return "".join([s for s in stack])
def calculate_points(completion: str) -> int:
points = 0
for char in completion:
points *= 5
points += POINTS2[char]
return points
def part_2(lines: list[str]) -> int:
incomplete = [line for line in lines if find_chunks(line) == 0]
points = []
for line in incomplete:
completion = complete_line(line)
points.append(calculate_points(completion))
points.sort()
return points[len(points) // 2]
if __name__ == "__main__":
path = init_parser()
lines = read_input(path)
print(f"Part 1: {part_1(lines)}")
print(f"Part 2: {part_2(lines)}")
def main(_):
raise NotImplementedError