-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathAoC2020_18.py
102 lines (79 loc) · 2.54 KB
/
AoC2020_18.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
#! /usr/bin/env python3
#
# Advent of Code 2020 Day 18
#
from typing import NamedTuple
from aoc import my_aocd
from aoc.common import log
def _tokenize(input_: str) -> list[str]:
return [_ for _ in input_ if _ != " "]
class ResultAndPosition(NamedTuple):
result: int
position: int
def _evaluate(expression: list[str], pos: int = 0) -> ResultAndPosition:
stack = [0, "+"]
result = 0
i = pos
while i < len(expression):
_ = expression[i]
if _ in {"+", "*"}:
stack.append(_)
elif _ == ")":
return ResultAndPosition(result, i)
else:
if _ == "(":
sub = _evaluate(expression, i+1)
operand_2 = sub.result
i = sub.position
elif not _.isnumeric():
raise ValueError("invalid input")
else:
operand_2 = int(_)
operator = stack.pop()
operand_1 = stack.pop()
if operator == "+":
result = operand_1 + operand_2
else:
result = operand_1 * operand_2
stack.append(result)
i += 1
return ResultAndPosition(result, i)
def part_1(inputs: tuple[str]) -> int:
return sum([_evaluate(_tokenize(input_)).result
for input_ in inputs
])
def _fix_for_addition_preference(expr: str) -> str:
x = expr.replace("(", "((")
x = x.replace(")", "))")
x = "(" + x.replace("*", ") * (") + ")"
log(f"{expr} -> {x}")
return x
def part_2(inputs: tuple[str]) -> int:
return sum([eval(_fix_for_addition_preference(input_)) # nosec
for input_ in inputs
])
def part_2_bis(inputs: tuple[str]) -> int:
return sum([_evaluate(_tokenize(
_fix_for_addition_preference(input_))).result
for input_ in inputs
])
TEST = """\
1 + 2 * 3 + 4 * 5 + 6
1 + (2 * 3) + (4 * (5 + 6))
2 * 3 + (4 * 5)
5 + (8 * 3 + 9 + 3 * 4 * 3)
5 * 9 * (7 * 3 * 3 + 9 * 3 + (8 + 6 * 4))
((2 + 4 * 9) * (6 + 9 * 8 + 6) + 6) + 2 + 4 * 2
""".splitlines()
def main() -> None:
my_aocd.print_header(2020, 18)
assert part_1(TEST) == 71 + 51 + 26 + 437 + 12240 + 13632
assert part_2(TEST) == 231 + 51 + 46 + 1445 + 669060 + 23340
assert part_2_bis(TEST) == 231 + 51 + 46 + 1445 + 669060 + 23340
inputs = my_aocd.get_input(2020, 18, 383)
result1 = part_1(inputs)
print(f"Part 1: {result1}")
result2 = part_2(inputs)
print(f"Part 2: {result2}")
if __name__ == '__main__':
main()