-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathprefill_cache.py
149 lines (115 loc) · 4.99 KB
/
prefill_cache.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
import dataclasses
import os.path
import pdb
from concurrent.futures import as_completed
import math
import queue
import sys
import shapely.geometry
from requests_futures.sessions import FuturesSession
from shapely.geometry import Point
from download_gpkg import SAN_FRAN, create_poly_from_geojson
url = sys.argv[1]
def lat_lon_to_tile(lat, lon, zoom):
x = (lon + 180) / 360 * (2 ** zoom)
lat_rad = math.radians(lat)
y = (1 - math.log(math.tan(lat_rad) + 1 / math.cos(lat_rad)) / math.pi) / 2 * (2 ** zoom)
return int(x), int(y)
# Generate the reverse code of lat_lon_to_tile that returns bounds of the tile in shapely format
def tile_to_lat_lon_bounds(x, y, zoom):
n = 2.0 ** zoom
lon_deg = x / n * 360.0 - 180.0
lat_rad = math.atan(math.sinh(math.pi * (1 - 2 * y / n)))
lat_deg = math.degrees(lat_rad)
n = 2.0 ** zoom
lon_deg2 = (x + 1) / n * 360.0 - 180.0
lat_rad2 = math.atan(math.sinh(math.pi * (1 - 2 * (y + 1) / n)))
lat_deg2 = math.degrees(lat_rad2)
return shapely.geometry.box(min(lon_deg, lon_deg2), min(lat_deg, lat_deg2), max(lon_deg2, lon_deg), max(lat_deg2, lat_deg))
@dataclasses.dataclass
class Explore:
zoom: int
x: int
y: int
@classmethod
def from_latlong(cls, lat: float, lng: float, zoom: int):
x, y = lat_lon_to_tile(lat, lng, zoom)
return Explore(zoom, x, y)
def __hash__(self):
return hash((self.zoom, self.x, self.y))
to_explore = queue.Queue()
VANCOUVER_TL = (49.328910726698005, -123.25959613136175)
VANCOUVER_BR = (49.15112007739324, -122.86397283811988)
MONTREAL_1 = (45.64534888679888, -73.5359544928739)
MONTREAL_2 = (45.430926507657276, -73.76828002313724)
NYC_1 = (40.64775151884592, -74.1134681369716)
NYC_2 = (40.94121411047778, -73.78936332225874)
TORONTO = (43.6532, -79.3832)
PARIS_1 = (48.91191595093818, 2.1980415114905725)
LONDON_1 = (51.672343, -0.148271)
SAN_FRANCISCO = (37.789407162468066, -122.35309872004174)
SAN_FRANCISCO1 = (37.50211486594995, -122.10646136267307)
SF2 = 37.7690145460696, -122.43082602680231
SF3 = 37.803398137952186, -122.22809341889861
CHICAGO = 41.88502620493033, -87.64866240164858
CHICAGO2 = 42.0965437845508, -87.73648980584558
# to_explore.put(Explore.from_latlong(*NYC_1, 7))
# to_explore.put(Explore.from_latlong(*NYC_2, 7))
# to_explore.put(Explore.from_latlong(*VANCOUVER_TL, 7))
# to_explore.put(Explore.from_latlong(*VANCOUVER_BR, 7))
# to_explore.put(Explore.from_latlong(*MONTREAL_1, 7))
# to_explore.put(Explore.from_latlong(*MONTREAL_2, 7))
# to_explore.put(Explore.from_latlong(*PARIS_1, 7))
# to_explore.put(Explore.from_latlong(*TORONTO, 7))
# to_explore.put(Explore.from_latlong(*SAN_FRANCISCO, 8))
# to_explore.put(Explore.from_latlong(*SAN_FRANCISCO1, 8))
to_explore.put(Explore.from_latlong(*SF3, 7))
to_explore.put(Explore.from_latlong(*SF2, 7))
# to_explore.put(Explore.from_latlong(*LONDON_1, 7))
to_explore_calculated = set()
MAX_ZOOM = 16
while not to_explore.empty():
explore = to_explore.get_nowait()
to_explore_calculated.add(explore)
# Calculate tiles to max zoom
for cur_zoom in range(explore.zoom + 1, MAX_ZOOM + 1):
zoom_diff = cur_zoom - explore.zoom
multiplier = int(math.pow(2, zoom_diff))
to_explore.put(Explore(cur_zoom, multiplier * explore.x, multiplier * explore.y))
to_explore.put(Explore(cur_zoom, multiplier * explore.x + 1, multiplier * explore.y))
to_explore.put(Explore(cur_zoom, multiplier * explore.x, multiplier * explore.y + 1))
to_explore.put(Explore(cur_zoom, multiplier * explore.x + 1, multiplier * explore.y + 1))
SF_POLY = create_poly_from_geojson(SAN_FRAN)
def pre_check(coord: Explore):
latlng = tile_to_lat_lon_bounds(coord.x, coord.y, coord.zoom)
if SF_POLY.intersects(latlng):
return os.path.exists(f"vancouver-cache/all_cities/{coord.zoom}/{coord.x}/{coord.y}.pbf")
else:
return True
# if os.path.exists(f"vancouver-cache/all_cities/{coord.zoom}/{coord.x}/{coord.y}.pbf"):
# os.remove(f"vancouver-cache/all_cities/{coord.zoom}/{coord.x}/{coord.y}.pbf")
# return True
with FuturesSession() as session:
futures = []
completed = 0
for coord in to_explore_calculated:
assert coord.zoom >= 7
if pre_check(coord):
completed += 1
continue
r = session.get(f'{url}/{coord.zoom}/{coord.x}/{coord.y}.pbf')
futures.append(r)
if len(futures) > 80:
for future in as_completed(futures):
resp = future.result()
# resp.raise_for_status()
if completed % 100 == 0:
print("Completed! ", resp, len(resp.content), f"{completed} out of {len(to_explore_calculated)}")
completed += 1
futures = []
for future in as_completed(futures):
resp = future.result()
resp.raise_for_status()
print("Completed! ", resp, len(resp.content), f"{completed} out of {len(to_explore_calculated)}")
completed += 1
session.executor.shutdown(wait = True)