-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgeocode.py
199 lines (166 loc) · 5.9 KB
/
geocode.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
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
import argparse
import glob
import concurrent.futures
import logging
import os
import threading
from typing import Any, Dict, Iterable, List
import exiftool
from geopy.geocoders import Nominatim
from geopy.extra.rate_limiter import RateLimiter
log = logging.getLogger(__name__)
exiftool_lock = threading.Lock()
# Dictionary of tags for ExifTool locations to data from geopy
# Made with the help of: https://github.com/OpenCageData/address-formatting/blob/master/conf/components.yaml
NOMINATIM_MAPPING = {
"MWG:Country": ["country", "country_name"],
"MWG:State": [
"state",
"province",
"region",
"island",
"state_code",
"state_district",
"county",
"county_code",
],
"MWG:City": [
"city",
"town",
"village",
"hamlet",
"locality",
"neighbourhood",
"suburb",
"city_district",
],
}
DEFAULT_USER_AGENT = "https://github.com/bezineb5/reverse-geocoder"
def _parse_arguments():
parser = argparse.ArgumentParser(
description="Simple reverse geocoding with geopy and exiftool"
)
parser.add_argument("files", nargs="+", help="files to reverse geocode")
parser.add_argument("-ua", required=False, default=DEFAULT_USER_AGENT)
return parser.parse_args()
def _map_nominatim_place_to_tags(raw_place: Dict[str, Any]) -> List[str]:
display_name = raw_place.get("display_name")
address = raw_place.get("address")
params: List[str] = []
for tag, v in NOMINATIM_MAPPING.items():
if isinstance(v, str):
# Single value
content = address.get(v)
if content:
params.append("-{tag}={content}".format(tag=tag, content=content))
else:
# List of values, by decrementing order of interest
for key in v:
content = address.get(key)
if content:
params.append("-{tag}={content}".format(tag=tag, content=content))
break
if display_name:
components = display_name.split(", ")
if components:
house_number = address.get("house_number")
if house_number and components[0] == house_number and len(components) > 1:
params.append(
"-{tag}={content}".format(
tag="MWG:Location", content=", ".join(components[0:2])
)
)
else:
params.append(
"-{tag}={content}".format(tag="MWG:Location", content=components[0])
)
return params
def reverse_geocode(et: exiftool.ExifToolHelper, reverse_geolocator, f: str):
# Find place names from GPS already in file
# Note: use the XMP tags so that lat/long has a - sign for W or S
log.info("Geocoding %s", f)
with exiftool_lock:
metada_output = et.get_tags(
f,
[
"XMP:GPSLatitude",
"XMP:GPSLongitude",
"XMP:Country",
"Composite:GPSLatitude",
"Composite:GPSLongitude",
],
)
if metada_output:
gps_dict = metada_output[0]
else:
log.info("No metadata found")
return
lat = gps_dict.get("XMP:GPSLatitude", gps_dict.get("Composite:GPSLatitude"))
lng = gps_dict.get("XMP:GPSLongitude", gps_dict.get("Composite:GPSLongitude"))
previous_country = gps_dict.get("XMP:Country")
if not lat or not lng:
log.info("Aborting, missing lat/long")
return
if previous_country:
log.info("Aborting, previous country: %s", previous_country)
return
location = reverse_geolocator((lat, lng))
if not location:
return
log.info(location.raw)
nominatim_address = location.raw
if not nominatim_address:
return
# Apply reverse geocoding using MWG tag standards
# (see, e.g., http://www.metadataworkinggroup.org/ and
# http://www.sno.phy.queensu.ca/~phil/exiftool/TagNames/MWG.html)
# This works best for cities, for rural areas (e.g. when hiking) let's
# use the county if available. Would be great to get more info, but for
# now that works okay. There may be a lot of variation in what makes
# sense, especially the Location tag, so may need to make this more
# sophisticated in the future.
params = _map_nominatim_place_to_tags(nominatim_address)
# Then add the filename and encode everything
params.append("-srcfile %d%f.xmp")
params.append("-overwrite_original")
params.append(f)
# Do the tagging! It seems that execute_json fails in that case.
bytes_params = map(os.fsencode, params)
with exiftool_lock:
log.info(et.execute(*bytes_params, raw_bytes=False))
def geocode_files(
files: Iterable[str],
user_agent: str = "https://github.com/bezineb5/reverse-geocoder",
):
all_files: List[str] = []
for file in files:
all_files.extend(glob.glob(file))
if not all_files:
return
# Use Nominatim, from OpenStreetMap, for reverse lookup
geolocator = Nominatim(user_agent=user_agent)
reverse_func = RateLimiter(
geolocator.reverse,
max_retries=1,
min_delay_seconds=1,
swallow_exceptions=False,
)
with exiftool.ExifToolHelper() as et:
def geocoding(f: str):
try:
reverse_geocode(et, reverse_func, f)
except Exception as e:
log.exception("Error while geotagging: %s", f)
return
with concurrent.futures.ThreadPoolExecutor(max_workers=2) as executor:
for f in all_files:
executor.submit(geocoding, f)
executor.shutdown()
def main():
logging.basicConfig(level=logging.INFO)
args = _parse_arguments()
if not args.files:
return
geocode_files(args.files, user_agent=args.ua)
if __name__ == "__main__":
main()