-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmonkey_in_the_middle.py
113 lines (77 loc) · 2.46 KB
/
monkey_in_the_middle.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
106
107
108
109
110
111
112
113
from utils import read_input, run
from dataclasses import dataclass
from collections import deque, defaultdict
from operator import add, mul
from typing import Callable
from functools import cache
FNAME = "11/input.txt"
LCM = 2 * 3 * 5 * 7 * 11 * 13 * 17 * 19
@dataclass
class Monkey:
id: int
items: deque[int]
operation: Callable[[int], int]
next_monkey: Callable[[int], int]
inspections: int = 0
def has_items(self):
return bool(self.items)
def next_item(self):
return self.items.popleft()
def inspect(self, item, is_part_one=True):
self.inspections += 1
worry = self.operation(item)
if is_part_one:
worry //= 3
else:
worry %= LCM
return worry, self.next_monkey(worry)
def recieve_item(self, item):
self.items.append(item)
OPERATORS = {
'+': add,
'*': mul
}
def parse_chunk(chunk):
lines = chunk.splitlines()
number = int(lines[0].split()[-1][:-1])
items = deque(map(int, lines[1][18:].split(', ')))
op_symbol, last = lines[2].split()[-2:]
op = OPERATORS[op_symbol]
if last == 'old':
variable = lambda old: old
else:
variable = lambda _: int(last)
def operation(old):
return op(old, variable(old))
test_divisible = int(lines[3].split()[-1])
true_monkey = int(lines[4].split()[-1])
false_monkey = int(lines[5].split()[-1])
def next_monkey(worry_level):
if worry_level % test_divisible == 0:
return true_monkey
return false_monkey
return Monkey(number, items, operation, next_monkey)
##########
# PART 1 #
##########
def round(monkeys, is_part_one=True):
for monkey in monkeys:
while monkey.has_items():
item = monkey.next_item()
item, next_monkey = monkey.inspect(item, is_part_one)
monkeys[next_monkey].recieve_item(item)
def part_one(input_file):
monkeys = read_input(input_file, separator='\n\n', parse_chunk=parse_chunk)
for _ in range(20):
round(monkeys)
return mul(*sorted(m.inspections for m in monkeys)[-2:])
##########
# PART 2 #
##########
def part_two(input_file):
monkeys = read_input(input_file, separator='\n\n', parse_chunk=parse_chunk)
for _ in range(10000):
round(monkeys, is_part_one=False)
return mul(*sorted(m.inspections for m in monkeys)[-2:])
if __name__ == '__main__':
run(part_one, part_two, FNAME)