__filename__ = "maps.py" __author__ = "Bob Mottram" __license__ = "AGPL3+" __version__ = "1.6.0" __maintainer__ = "Bob Mottram" __email__ = "bob@libreserver.org" __status__ = "Production" __module_group__ = "Core" import os from flags import is_float from utils import browser_supports_download_filename from utils import get_url_from_post from utils import acct_dir from utils import load_json from utils import save_json from utils import locate_post from utils import remove_html from utils import has_object_dict from utils import date_utcnow from utils import date_epoch from utils import date_from_string_format from session import get_resolved_url def _geocoords_to_osm_link(osm_domain: str, zoom: int, latitude: float, longitude: float) -> str: """Returns an OSM link for the given geocoordinates """ return 'https://www.' + osm_domain + '/#map=' + \ str(zoom) + '/' + str(latitude) + '/' + str(longitude) def get_location_dict_from_tags(tags: []) -> str: """Returns the location from the tags list """ for tag_item in tags: if not tag_item.get('type'): continue if tag_item['type'] != 'Place': continue if not tag_item.get('name'): continue if not isinstance(tag_item['name'], str): continue return tag_item return None def _get_location_from_tags(tags: []) -> str: """Returns the location from the tags list """ locn = get_location_dict_from_tags(tags) if locn: location_str = locn['name'].replace('\n', ' ') return remove_html(location_str) return None def get_location_from_post(post_json_object: {}) -> str: """Returns the location for the given post """ locn = None # location represented via a tag post_obj = post_json_object if has_object_dict(post_json_object): post_obj = post_json_object['object'] if post_obj.get('tag'): if isinstance(post_obj['tag'], list): locn = _get_location_from_tags(post_obj['tag']) # location representation used by pixelfed locn_exists = False locn2 = None if post_obj.get('location'): locn2 = post_obj['location'] if isinstance(locn2, dict): if locn2.get('longitude') and \ locn2.get('latitude'): if isinstance(locn2['longitude'], str) and \ isinstance(locn2['latitude'], str): if is_float(locn2['longitude']) and \ is_float(locn2['latitude']): locn_exists = True if not locn_exists: if locn2.get('name'): if isinstance(locn2['name'], str): locn = locn2['name'] if locn_exists: osm_domain = 'osm.org' zoom = 17 locn = _geocoords_to_osm_link(osm_domain, zoom, locn2['latitude'], locn2['longitude']) return locn def _geocoords_from_osm_link(url: str, osm_domain: str) -> (int, float, float): """Returns geocoordinates from an OSM map link """ if osm_domain not in url: return None, None, None if '#map=' not in url: return None, None, None coords_str = url.split('#map=')[1] if '/' not in coords_str: return None, None, None coords = coords_str.split('/') if len(coords) != 3: return None, None, None zoom = coords[0] if not zoom.isdigit(): return None, None, None latitude = coords[1] if not is_float(latitude): return None, None, None longitude = coords[2] if not is_float(longitude): return None, None, None zoom = int(zoom) latitude = float(latitude) longitude = float(longitude) return zoom, latitude, longitude def _geocoords_from_osmorg_link(url: str) -> (int, float, float): """Returns geocoordinates from an OSM map link """ osm_domain = 'osm.org' if osm_domain not in url: return None, None, None if 'mlat=' not in url: return None, None, None if 'mlon=' not in url: return None, None, None if 'zoom=' not in url and '#map=' not in url: return None, None, None latitude = url.split('mlat=')[1] if '&' in latitude: latitude = latitude.split('&')[0] if not is_float(latitude): return None, None, None longitude = url.split('mlon=')[1] if '&' in longitude: longitude = longitude.split('&')[0] if not is_float(longitude): return None, None, None if 'zoom=' in url: zoom = url.split('zoom=')[1] if '&' in zoom: zoom = zoom.split('&')[0] else: zoom = url.split('#map=')[1] if '/' in zoom: zoom = zoom.split('/')[0] if not zoom.isdigit(): return None, None, None zoom = int(zoom) latitude = float(latitude) longitude = float(longitude) return zoom, latitude, longitude def _geocoords_from_osmorg_go_link(url: str, session) -> (int, float, float): """Returns geocoordinates from an OSM go map link """ osm_domain = 'osm.org' if osm_domain not in url: return None, None, None if 'mlat=' in url: return None, None, None if 'mlon=' in url: return None, None, None if '/go/' not in url: return None, None, None # resolve url equivalent to # curl -Ls -o /dev/null -w %{url_effective} [url] resolved_url = get_resolved_url(session, url) if not resolved_url: return None, None, None if 'osm.org' in resolved_url: (zoom, latitude, longitude) = \ _geocoords_from_osmorg_link(resolved_url) else: (zoom, latitude, longitude) = \ _geocoords_from_osm_link(resolved_url, 'openstreetmap.org') return zoom, latitude, longitude def _geocoords_from_osmand_link(url: str) -> (int, float, float): """Returns geocoordinates from an OSM android map link """ latitude = None longitude = None zoom = 10 if 'pin=' in url: pin_coords_str = url.split('pin=')[1] if ',' in pin_coords_str: latitude_str = pin_coords_str.split(',')[0] longitude_str = pin_coords_str.split(',')[1] if is_float(latitude_str) and is_float(longitude_str): latitude = float(latitude_str) longitude = float(longitude_str) if '#' in url: coords_str = url.split('#')[1] if '/' in coords_str: sections = coords_str.split('/') if len(sections) == 3: zoom_str = sections[0] latitude_str = sections[1] longitude_str = sections[2] if zoom_str.isnumeric() and \ is_float(latitude_str) and \ is_float(longitude_str): latitude = float(latitude_str) longitude = float(longitude_str) zoom = int(zoom_str) return zoom, latitude, longitude def _geocoords_from_geo_link(url: str) -> (int, float, float): """Returns geocoordinates from an geo link https://en.wikipedia.org/wiki/Geo_URI_scheme """ latitude = None longitude = None zoom = 10 coords_str = url.split('geo:')[1] if ',' in coords_str: coords_sections = coords_str.split(',') if len(coords_sections) >= 2: latitude_str = coords_sections[0] longitude_str = coords_sections[1] if ';' in longitude_str: longitude_str = longitude_str.split(';')[0] if '?' in longitude_str: longitude_str = longitude_str.split('?')[0] if ' ' in longitude_str: longitude_str = longitude_str.split(' ')[0] if is_float(latitude_str) and is_float(longitude_str): latitude = float(latitude_str) longitude = float(longitude_str) return zoom, latitude, longitude def _geocoords_from_gmaps_link(url: str) -> (int, float, float): """Returns geocoordinates from a Gmaps link """ if '/maps/' not in url: return None, None, None coords_str = url.split('/maps', 1)[1] if '/@' not in coords_str: return None, None, None coords_str = coords_str.split('/@', 1)[1] if 'z' not in coords_str: return None, None, None coords_str = coords_str.split('z', 1)[0] if ',' not in coords_str: return None, None, None coords = coords_str.split(',') if len(coords) != 3: return None, None, None zoom = coords[2] if not zoom.isdigit(): if is_float(str(zoom)): zoom = int(float(str(zoom))) else: return None, None, None latitude = coords[0] if not is_float(latitude): return None, None, None longitude = coords[1] if not is_float(longitude): return None, None, None zoom = int(zoom) latitude = float(latitude) longitude = float(longitude) return zoom, latitude, longitude def _geocoords_from_bmaps_link(url: str) -> (int, float, float): """Returns geocoordinates from a bing map link """ prefixes = ('/maps?cp=', '/maps/directions?cp=') map_prefix = None for prefix in prefixes: if prefix in url: map_prefix = prefix break if not map_prefix: return None, None, None coords_str = url.split(map_prefix)[1] if '~' not in coords_str: return None, None, None orig_coords_str = coords_str if '&' in coords_str: coords_str = coords_str.split('&')[0] if ';' in coords_str: coords_str = coords_str.split(';')[0] coords = coords_str.split('~') if len(coords) != 2: return None, None, None latitude = coords[0] if not is_float(latitude): return None, None, None longitude = coords[1] if not is_float(longitude): return None, None, None zoom = 17 if 'lvl=' in orig_coords_str: zoom = orig_coords_str.split('lvl=')[1] if '&' in zoom: zoom = zoom.split('&')[0] if ';' in zoom: zoom = zoom.split(';')[0] if not zoom.isdigit(): return None, None, None zoom = int(zoom) latitude = float(latitude) longitude = float(longitude) return zoom, latitude, longitude def _geocoords_from_waze_link(url: str) -> (int, float, float): """Returns geocoordinates from a waze map link """ prefixes = ['/ul?ll='] map_prefix = None for prefix in prefixes: if prefix in url: map_prefix = prefix break if not map_prefix: return None, None, None coords_str = url.split(map_prefix)[1] orig_coords_str = coords_str if '&' in coords_str: coords_str = coords_str.split('&')[0] if '%2C' not in coords_str and ',' not in coords_str: return None, None, None if '%2C' in coords_str: coords = coords_str.split('%2C') else: coords = coords_str.split(',') if len(coords) != 2: return None, None, None latitude = coords[0] if not is_float(latitude): return None, None, None longitude = coords[1] if not is_float(longitude): return None, None, None zoom = 17 if 'zoom=' in orig_coords_str: zoom = orig_coords_str.split('zoom=')[1] if '&' in zoom: zoom = zoom.split('&')[0] if not zoom.isdigit(): return None, None, None zoom = int(zoom) latitude = float(latitude) longitude = float(longitude) return zoom, latitude, longitude def _geocoords_from_wego_link(url: str) -> (int, float, float): """Returns geocoordinates from a wego map link """ prefixes = ['/?map='] map_prefix = None for prefix in prefixes: if prefix in url: map_prefix = prefix break if not map_prefix: return None, None, None coords_str = url.split(map_prefix)[1] if ',' not in coords_str: return None, None, None coords = coords_str.split(',') if len(coords) < 3: return None, None, None latitude = coords[0] if not is_float(latitude): return None, None, None longitude = coords[1] if not is_float(longitude): return None, None, None zoom = coords[2] if not zoom.isdigit(): return None, None, None zoom = int(zoom) latitude = float(latitude) longitude = float(longitude) return zoom, latitude, longitude def geocoords_from_map_link(url: str, osm_domain: str, session) -> (int, float, float): """Returns geocoordinates from a map link url """ if osm_domain in url: zoom, latitude, longitude = \ _geocoords_from_osm_link(url, osm_domain) return zoom, latitude, longitude if 'osm.org' in url and 'mlat=' not in url and '/go/' in url: zoom, latitude, longitude = \ _geocoords_from_osmorg_go_link(url, session) return zoom, latitude, longitude if 'osm.org' in url and 'mlat=' in url: zoom, latitude, longitude = \ _geocoords_from_osmorg_link(url) return zoom, latitude, longitude if 'osmand.net' in url and '/map' in url: zoom, latitude, longitude = \ _geocoords_from_osmand_link(url) return zoom, latitude, longitude if '.google.co' in url: zoom, latitude, longitude = \ _geocoords_from_gmaps_link(url) return zoom, latitude, longitude if '.bing.co' in url: zoom, latitude, longitude = \ _geocoords_from_bmaps_link(url) return zoom, latitude, longitude if '.waze.co' in url: zoom, latitude, longitude = \ _geocoords_from_waze_link(url) return zoom, latitude, longitude if 'wego.here.co' in url: zoom, latitude, longitude = \ _geocoords_from_wego_link(url) return zoom, latitude, longitude if 'geo:' in url and ',' in url: zoom, latitude, longitude = \ _geocoords_from_geo_link(url) return zoom, latitude, longitude return None, None, None def html_open_street_map(url: str, bounding_box_degrees: float, translate: {}, session, session_onion, session_i2p, width: str = "725", height: str = "650") -> str: """Returns embed html for an OSM link """ osm_domain = 'openstreetmap.org' map_session = session if '.onion/' in url: map_session = session_onion elif '.i2p/' in url: map_session = session_i2p zoom, latitude, longitude = \ geocoords_from_map_link(url, osm_domain, map_session) if not latitude: return '' if not longitude: return '' if not zoom: return '' osm_url = _geocoords_to_osm_link(osm_domain, zoom, latitude, longitude) html_str = \ '
' + translate['View Larger Map'] + '\n' return html_str def set_map_preferences_url(base_dir: str, nickname: str, domain: str, maps_website_url: str) -> None: """Sets the preferred maps website for an account """ maps_filename = \ acct_dir(base_dir, nickname, domain) + '/map_preferences.json' if os.path.isfile(maps_filename): maps_json = load_json(maps_filename) maps_json['url'] = maps_website_url else: maps_json = { 'url': maps_website_url } save_json(maps_json, maps_filename) def get_map_preferences_url(base_dir: str, nickname: str, domain: str) -> str: """Gets the preferred maps website for an account """ maps_filename = \ acct_dir(base_dir, nickname, domain) + '/map_preferences.json' if os.path.isfile(maps_filename): maps_json = load_json(maps_filename) if maps_json.get('url'): url_str = get_url_from_post(maps_json['url']) return remove_html(url_str) return None def set_map_preferences_coords(base_dir: str, nickname: str, domain: str, latitude: float, longitude: float, zoom: int) -> None: """Sets the preferred maps website coordinates for an account """ maps_filename = \ acct_dir(base_dir, nickname, domain) + '/map_preferences.json' if os.path.isfile(maps_filename): maps_json = load_json(maps_filename) maps_json['latitude'] = latitude maps_json['longitude'] = longitude maps_json['zoom'] = zoom else: maps_json = { 'latitude': latitude, 'longitude': longitude, 'zoom': zoom } save_json(maps_json, maps_filename) def get_map_preferences_coords(base_dir: str, nickname: str, domain: str) -> (float, float, int): """Gets the preferred maps website coordinates for an account """ maps_filename = \ acct_dir(base_dir, nickname, domain) + '/map_preferences.json' if os.path.isfile(maps_filename): maps_json = load_json(maps_filename) if maps_json.get('latitude') and \ maps_json.get('longitude') and \ maps_json.get('zoom'): return maps_json['latitude'], \ maps_json['longitude'], \ maps_json['zoom'] return None, None, None def get_map_links_from_post_content(content: str, session) -> []: """Returns a list of map links """ osm_domain = 'openstreetmap.org' sections = content.split('://') map_links: list[str] = [] ctr = 0 for link_str in sections: if ctr == 0: ctr += 1 continue url = link_str if '"' in url: url = url.split('"')[0] if '<' in url: url = url.split('<')[0] if not url: continue # complete the url if 'http://' + url in content: url = 'http://' + url elif 'https://' + url in content: url = 'https://' + url zoom, latitude, longitude = \ geocoords_from_map_link(url, osm_domain, session) if not latitude: continue if not longitude: continue if not zoom: continue if url not in map_links: map_links.append(url) ctr += 1 # https://en.wikipedia.org/wiki/Geo_URI_scheme ctr = 0 sections = content.split('geo:') for link_str in sections: if ctr == 0: ctr += 1 continue if ',' not in link_str: continue coords_str = '' for char in link_str: if not char.isnumeric() and char not in (',', '-', '.'): break coords_str += char if ',' not in coords_str: continue coord_sections = coords_str.split(',') if len(coord_sections) < 2: continue if not is_float(coord_sections[0]) or \ not is_float(coord_sections[1]): continue url = 'geo:' + coords_str map_links.append(url) ctr += 1 return map_links def add_tag_map_links(tag_maps_dir: str, tag_name: str, map_links: [], published: str, post_url: str) -> None: """Appends to a hashtag file containing map links This is used to show a map for a particular hashtag """ tag_map_filename = tag_maps_dir + '/' + tag_name + '.txt' post_url = post_url.replace('#', '/') # read the existing map links existing_map_links: list[str] = [] if os.path.isfile(tag_map_filename): try: with open(tag_map_filename, 'r', encoding='utf-8') as fp_tag: existing_map_links = fp_tag.read().split('\n') except OSError: print('EX: error reading tag map ' + tag_map_filename) # combine map links with the existing list secs_since_epoch = \ int((date_from_string_format(published, ['%Y-%m-%dT%H:%M:%S%z']) - date_epoch()).total_seconds()) links_changed = False for link in map_links: line = str(secs_since_epoch) + ' ' + link + ' ' + post_url if line in existing_map_links: continue links_changed = True existing_map_links = [line] + existing_map_links if not links_changed: return # sort the list of map links existing_map_links.sort(reverse=True) map_links_str = '' ctr = 0 for link in existing_map_links: if not link: continue map_links_str += link + '\n' ctr += 1 # don't allow the list to grow indefinitely if ctr >= 2000: break # save the tag try: with open(tag_map_filename, 'w+', encoding='utf-8') as fp_tag: fp_tag.write(map_links_str) except OSError: print('EX: error writing tag map ' + tag_map_filename) def _gpx_location(latitude: float, longitude: float, post_id: str) -> str: """Returns a gpx waypoint """ map_str = '\n' map_str += ' ' + post_id + '\n' map_str += ' \n' map_str += '\n' return map_str def _kml_location(place_ctr: int, latitude: float, longitude: float, post_id: str) -> str: """Returns a kml placemark """ map_str = '\n' map_str += ' ' + str(place_ctr) + '\n' map_str += ' ' + \ post_id + '\n]]>\n' map_str += ' \n' map_str += ' \n' map_str += ' ' + str(longitude) + ',' + \ str(latitude) + ',0\n' map_str += ' \n' map_str += '\n' return map_str def _hashtag_map_to_format(base_dir: str, tag_name: str, start_hours_since_epoch: int, end_hours_since_epoch: int, nickname: str, domain: str, map_format: str, session) -> str: """Returns the KML/GPX for a given hashtag between the given times """ place_ctr = 0 osm_domain = 'openstreetmap.org' tag_map_filename = base_dir + '/tagmaps/' + tag_name + '.txt' if map_format == 'gpx': map_str = '\n' map_str += '\n' else: map_str = '\n' map_str += '\n' map_str += '\n' if os.path.isfile(tag_map_filename): map_links: list[str] = [] try: with open(tag_map_filename, 'r', encoding='utf-8') as fp_tag: map_links = fp_tag.read().split('\n') except OSError: print('EX: unable to read tag map links ' + tag_map_filename) if map_links: start_secs_since_epoch = int(start_hours_since_epoch * 60 * 60) end_secs_since_epoch = int(end_hours_since_epoch * 60 * 60) for link_line in map_links: link_line = link_line.strip().split(' ') if len(link_line) < 3: continue # is this geocoordinate within the time range? secs_since_epoch = int(link_line[0]) if secs_since_epoch < start_secs_since_epoch or \ secs_since_epoch > end_secs_since_epoch: continue # get the geocoordinates from the map link map_link = link_line[1] zoom, latitude, longitude = \ geocoords_from_map_link(map_link, osm_domain, session) if not zoom: continue if not latitude: continue if not longitude: continue post_id = link_line[2] # check if the post is muted, and exclude the # geolocation if it is if nickname: post_filename = \ locate_post(base_dir, nickname, domain, post_id) if post_filename: if os.path.isfile(post_filename + '.muted'): continue place_ctr += 1 if map_format == 'gpx': map_str += _gpx_location(latitude, longitude, post_id) else: map_str += \ _kml_location(place_ctr, latitude, longitude, post_id) if map_format == 'gpx': map_str += '' else: map_str += '\n' map_str += '' if place_ctr == 0: return None return map_str def _hashtag_map_within_hours(base_dir: str, tag_name: str, hours: int, map_format: str, nickname: str, domain: str, session) -> str: """Returns gpx/kml for a hashtag containing maps for the last number of hours """ secs_since_epoch = \ int((date_utcnow() - date_epoch()).total_seconds()) curr_hours_since_epoch = int(secs_since_epoch / (60 * 60)) start_hours_since_epoch = curr_hours_since_epoch - abs(hours) end_hours_since_epoch = curr_hours_since_epoch + 2 map_str = \ _hashtag_map_to_format(base_dir, tag_name, start_hours_since_epoch, end_hours_since_epoch, nickname, domain, map_format, session) return map_str def _get_tagmaps_time_periods() -> {}: """dict of time periods for map display """ return { "Last hour": -1, "Last 3 hours": -3, "Last 6 hours": -6, "Last 12 hours": -12, "Last day": -24, "Last 2 days": -48, "Last week": -24 * 7, "Last 2 weeks": -24 * 7 * 2, "Last month": -24 * 7 * 4, "Last 6 months": -24 * 7 * 4 * 6, "Last year": -24 * 7 * 4 * 12 } def map_format_from_tagmaps_path(base_dir: str, path: str, map_format: str, domain: str, session) -> str: """Returns gpx/kml for a given tagmaps path /tagmaps/tagname-time_period """ if '/tagmaps/' not in path: return None time_period = _get_tagmaps_time_periods() tag_name = path.split('/tagmaps/')[1] if '-' in tag_name: tag_name = tag_name.split('-')[0] if not tag_name: return None for period_str, hours in time_period.items(): period_str2 = period_str.replace('Last ', '').lower() endpoint_str = \ '/tagmaps/' + tag_name + '-' + period_str2.replace(' ', '_') if path != endpoint_str: continue nickname = None if '/users/' in path: nickname = path.split('/users/')[1] if '/' in nickname: nickname = nickname.split('/')[0] return _hashtag_map_within_hours(base_dir, tag_name, hours, map_format, nickname, domain, session) return None def html_hashtag_maps(base_dir: str, tag_name: str, translate: {}, map_format: str, nickname: str, domain: str, session, ua_str: str) -> str: """Returns html for maps associated with a hashtag """ tag_map_filename = base_dir + '/tagmaps/' + tag_name + '.txt' if not os.path.isfile(tag_map_filename): return '' time_period = _get_tagmaps_time_periods() html_str = '' map_str = None ua_str_lower = ua_str.lower() for period_str, hours in time_period.items(): new_map_str = \ _hashtag_map_within_hours(base_dir, tag_name, hours, map_format, nickname, domain, session) if not new_map_str: continue if new_map_str == map_str: continue map_str = new_map_str period_str2 = period_str.replace('Last ', '').lower() tag_name_str = tag_name + '-' + period_str2.replace(' ', '_') endpoint_str = '/tagmaps/' + tag_name_str if html_str: html_str += ' ' description = period_str if translate.get(period_str): description = translate[period_str] if browser_supports_download_filename(ua_str_lower): html_str += '' + \ description + '' else: # NOTE: don't use download="preferredfilename" which is # unsupported by some browsers html_str += '' + \ description + '' if html_str: html_str = '📌 ' + html_str return html_str