-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathprepare_geojson.py
executable file
·162 lines (134 loc) · 4.99 KB
/
prepare_geojson.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
151
152
153
154
155
156
157
158
159
160
161
162
#!/usr/bin/env python3
"""Radlkarte GeoJSON Preparation Script
This script deals with GeoJSON files produced by JOSM,
i.e. a FeatureCollection with Points and LineStrings,
and serves these purposes:
1) reduce file size (for faster download)
2) calculate the bbox
3) stable feature order for minimum diffs after changes
(JOSM unfortunately reorders the GeoJSON)
For the latter point it adds unique ids to each feature and gracefully
handles duplicate and invalid ids.
"""
import json
import logging
import sys
logFormatter = "%(asctime)s - %(levelname)s - %(message)s"
logging.basicConfig(format=logFormatter, level=logging.INFO)
def enforce_int_id_in_feature_properties(feature):
if feature["properties"] is None:
feature["properties"] = {}
int_id = -1
try:
int_id = int(feature["properties"]["id"])
except KeyError:
logging.info("no id in {}".format(feature["properties"]))
except ValueError:
logging.info("invalid id in {}".format(feature["properties"]))
feature["properties"]["id"] = int_id
def restrict_decimal_precision(geometry):
tuples = None
if geometry["type"] == "LineString":
tuples = geometry["coordinates"]
elif geometry["type"] == "Point":
tuples = [geometry["coordinates"]]
else:
logging.warning("ignoring geometry type {}".format(geometry["type"]))
for tuple in tuples:
tuple[0] = float("{:.5f}".format(tuple[0]).rstrip("0").rstrip("."))
tuple[1] = float("{:.5f}".format(tuple[1]).rstrip("0").rstrip("."))
def get_max_id(features):
ids = [feature["properties"]["id"] for feature in features]
return max(ids)
def get_features_with_duplicate_or_invalid_ids(features):
seen_ids = set()
bad_features = []
for feature in features:
id = feature["properties"]["id"]
if id in seen_ids or id < 0:
bad_features.append(feature)
seen_ids.add(id)
return bad_features
def set_new_ids(features, start_id):
current_id = start_id
for feature in features:
feature["properties"]["id"] = current_id
current_id += 1
def calc_bbox(features):
"""
Simple bbox calculation, returns a list of min_lon, min_lat, max_lon, max_lat.
Won't work for datasets crossing the 180th meridian.
"""
min_lat = 90
max_lat = -90
min_lon = 180
max_lon = -180
for feature in features:
geometry = feature["geometry"]
tuples = None
if geometry["type"] == "LineString":
tuples = geometry["coordinates"]
elif geometry["type"] == "Point":
tuples = [geometry["coordinates"]]
else:
logging.warning("ignoring geometry type {}".format(geometry["type"]))
continue
lats = set()
lons = set()
for tuple in tuples:
lats.add(tuple[1])
lons.add(tuple[0])
if min(lats) < min_lat:
min_lat = min(lats)
if max(lats) > max_lat:
max_lat = max(lats)
if min(lons) < min_lon:
min_lon = min(lons)
if max(lons) > max_lon:
max_lon = max(lons)
return [min_lon, min_lat, max_lon, max_lat]
def minimize(infile, outfile):
data = None
try:
with open(infile, encoding=sys.getfilesystemencoding()) as json_file:
data = json.load(json_file)
except json.JSONDecodeError:
logging.warning("{} is not a valid json file - skipping.".format(infile))
return
if "features" not in data:
logging.warning("{} is not a valid geojson file - skipping.".format(infile))
return
features = data["features"]
logging.info("{} features parsed from {}".format(len(features), infile))
for feature in features:
enforce_int_id_in_feature_properties(feature)
restrict_decimal_precision(feature["geometry"])
bad_features = get_features_with_duplicate_or_invalid_ids(features)
max_id = max(1, get_max_id(features))
logging.info("max id found was {}".format(max_id))
set_new_ids(bad_features, max_id + 1)
id_to_feature = {feature["properties"]["id"]: feature for feature in features}
sorted_ids = sorted(id_to_feature.keys())
bbox = calc_bbox(features)
with open(outfile, "w") as json_file:
json_file.write(
f'{{"type": "FeatureCollection", "bbox": {bbox}, "features": [\n'
)
for id in sorted_ids[:-1]:
json_file.write(json.dumps(id_to_feature[id], sort_keys=True, indent=None))
json_file.write(",\n")
json_file.write(
json.dumps(id_to_feature[sorted_ids[-1]], sort_keys=True, indent=None)
)
json_file.write("]}")
logging.info(
"{} features ({} with new id) written to {}".format(
len(sorted_ids), len(bad_features), outfile
)
)
if __name__ == "__main__":
if len(sys.argv) > 1:
for infile in sys.argv[1:]:
minimize(infile, infile)
else:
print("Usage: one or more geojson files to be minimized in-place as arguments")