-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathday15.py
104 lines (85 loc) Β· 4.03 KB
/
day15.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
from dataclasses import dataclass
from functools import reduce
from shapely import LineString, Polygon, difference, geometry
from aocd import get_data
from parse import parse
@dataclass(kw_only=True, frozen=True)
class Point:
X: int
Y: int
@dataclass()
class Scan:
Sensor: Point
Beacon: Point
def distance(self):
return abs(self.Sensor.X - self.Beacon.X) + abs(self.Sensor.Y - self.Beacon.Y)
def get_points(self):
current_distance = self.distance()
points = [
geometry.Point(self.Sensor.X - current_distance, self.Sensor.Y),
geometry.Point(self.Sensor.X, self.Sensor.Y - current_distance),
geometry.Point(self.Sensor.X + current_distance, self.Sensor.Y),
geometry.Point(self.Sensor.X, self.Sensor.Y + current_distance),
geometry.Point(self.Sensor.X - current_distance, self.Sensor.Y)
]
return points
def get_polygon(self):
return geometry.Polygon([[p.x, p.y] for p in self.get_points()])
def read_scan_inputs(input_lines):
details = []
for line in input_lines:
parse_result = parse(input_line_template, line).fixed
new_scan = Scan(Point(X=parse_result[0], Y=parse_result[1]), Point(X=parse_result[2], Y=parse_result[3]))
details.append(new_scan)
return details
def part1(intersection_line, scan_inputs):
beacons_on_line = set()
scanners_on_line = set()
segments = set()
for scanner in scan_inputs:
if scanner.Beacon.Y == target_y_line:
beacons_on_line.add(scanner.Beacon)
if scanner.Sensor.Y == target_y_line:
scanners_on_line.add(scanner.Sensor)
current_poly = scanner.get_polygon()
intersection = current_poly.intersection(intersection_line)
if intersection.wkt != 'LINESTRING Z EMPTY':
x_coords = [int(intersection.bounds[0]), int(intersection.bounds[2])]
x_coords.sort()
segments = segments.union(set(range(x_coords[0], x_coords[1] + 1)))
return len(segments) - len(beacons_on_line)
if __name__ == '__main__':
data = [
'Sensor at x=2, y=18: closest beacon is at x=-2, y=15',
'Sensor at x=9, y=16: closest beacon is at x=10, y=16',
'Sensor at x=13, y=2: closest beacon is at x=15, y=3',
'Sensor at x=12, y=14: closest beacon is at x=10, y=16',
'Sensor at x=10, y=20: closest beacon is at x=10, y=16',
'Sensor at x=14, y=17: closest beacon is at x=10, y=16',
'Sensor at x=8, y=7: closest beacon is at x=2, y=10',
'Sensor at x=2, y=0: closest beacon is at x=2, y=10',
'Sensor at x=0, y=11: closest beacon is at x=2, y=10',
'Sensor at x=20, y=14: closest beacon is at x=25, y=17',
'Sensor at x=17, y=20: closest beacon is at x=21, y=22',
'Sensor at x=16, y=7: closest beacon is at x=15, y=3',
'Sensor at x=14, y=3: closest beacon is at x=15, y=3',
'Sensor at x=20, y=1: closest beacon is at x=15, y=3'
]
input_line_template = "Sensor at x={:d}, y={:d}: closest beacon is at x={:d}, y={:d}"
data = get_data(day=15, year=2022).splitlines()
scan_details = read_scan_inputs(data)
target_y_line = 2_000_000
line_infinity_bound = 10000000
draw_line = LineString(
[geometry.Point(line_infinity_bound * -1, target_y_line), geometry.Point(line_infinity_bound, target_y_line)])
print(f'Part 1: {part1(draw_line, scan_details)}')
multiplier = 4_000_000
searches = multiplier
# Build all the polygons
polygons = [scanner.get_polygon() for scanner in scan_details]
# Reduce applies a function to arguments
# Then do a set different between the Polygons and this polygon which is one giant rectangle
# because there will be one point that is missing that, then get that center as it is between the intersections
found_beacon = reduce(difference, polygons,
Polygon(((0, 0), (0, multiplier), (multiplier, multiplier), (multiplier, 0)))).centroid
print(f'Part 2: {(found_beacon.x * multiplier) + found_beacon.y}')