-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathday22.py
150 lines (116 loc) · 3.82 KB
/
day22.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
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
# ruff: noqa
import argparse
import os
from dataclasses import dataclass
from functools import cache, reduce
from itertools import product
class Grid:
"""Straight forward Grid model used for part 1"""
def __init__(self):
self.__on_nodes = set()
@property
def cubes_on(self) -> int:
return len(self.__on_nodes)
def switch(self, x_range: range, y_range: range, z_range: range, on: bool = True):
if any(
[
x_range.start < -50,
x_range.stop > 51,
y_range.start < -50,
y_range.stop > 51,
z_range.start < -50,
z_range.stop > 51,
]
):
return
if on:
for node in product(x_range, y_range, z_range, repeat=1):
self.__on_nodes.add(node)
else:
for node in product(x_range, y_range, z_range, repeat=1):
self.__on_nodes.discard(node)
def part1(input_) -> int:
grid = Grid()
for instruction, x, y, z in input_:
grid.switch(x, y, z, instruction == "on")
# print(f"Instruction Executed: {instruction} {x} {y} {z}")
return grid.cubes_on
# Start of part 2 code
@dataclass(frozen=True)
class Cuboid:
on: bool
x: range
y: range
z: range
def __repr__(self):
return (
f"Cuboid({'on' if self.on else 'off'}: x=[{self.x.start}..{self.x.stop - 1}] "
f"y=[{self.y.stop}..{self.y.stop - 1}] z=[{self.z.start}..{self.z.stop - 1}])"
)
@property
def magnitude(self) -> int:
return len(self.x) * len(self.y) * len(self.z)
def cuboid_intersect(c1: Cuboid, c2: Cuboid) -> Cuboid | None:
def intersect_1d(a, b):
start = max(a.start, b.start)
stop = min(a.stop, b.stop)
if start < stop:
return range(start, stop)
return None
x = intersect_1d(c1.x, c2.x)
y = intersect_1d(c1.y, c2.y)
z = intersect_1d(c1.z, c2.z)
if any([a is None for a in (x, y, z)]):
return None
return Cuboid(c1.on, x, y, z)
def intersects(c1: Cuboid, c2: Cuboid) -> bool:
return cuboid_intersect(c1, c2) is not None
@cache
def intersect_all(cuboids: set[Cuboid]) -> Cuboid | None:
def reduce_cuboids(a, b):
if a is None or b is None:
return None
return cuboid_intersect(a, b)
return reduce(reduce_cuboids, cuboids)
def part2(cuboids: set[Cuboid]) -> int:
"""
Inclusion-exclusion principle:
|Union of sets|=∑|Singletons|−∑|Pairs|+∑|Triples|−∑|Quadruples|+...+(−1)n+1|n-tuples|
"""
return 0
# Inclusion-exclusion principle for finding common points in sets
# or sweep line algo
# Input reading
def make_range(input_: str) -> range:
x = input_.split("=")[1]
a, b = x.split("..")
a = int(a)
b = int(b)
start = min(a, b)
end = max(a, b)
return range(int(start), int(end) + 1)
def read_input(filepath: str) -> list[str]:
with open(filepath, "r") as f:
for line in f.readlines():
on_off, ranges = line.strip().split()
dims = ranges.split(",")
x = make_range(dims[0])
y = make_range(dims[1])
z = make_range(dims[2])
yield on_off, x, y, z
def init_parser() -> str:
parser = argparse.ArgumentParser(description="Advent of Code day 22 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])
if __name__ == "__main__":
path = init_parser()
print(f"Part 1: {part1(read_input(path))}")
print(
f"Part 2: {part2({Cuboid(i == 'on', x, y, z) for i, x, y, z in read_input(path)})}"
)
print("Part 2: 2758514936282235 (test)")
def main(_):
raise NotImplementedError