Speedcams: Convert iGo (txt) to Sygic (dat)
Sygic (https://www.sygic.com) is my favorite GPS navigator. I use it often on my Android to protect myself against the oppression of speed cameras! The speedcam feature will show you notifications about known locations where you could be getting a speeding ticket.
For static speed cameras, i.e. those which are permanently installed next to roads, their database is included with the app and these will be shown even in offline mode, which is nice!
The problem is that the offline camera database is not often updated, especially for regions other than the North America and Europe.
To solve this problem we can download more up to date speedcam updates from 3rd party sites, such as https://www.maparadar.com (Brazil), or https://www.speedcams.eu (Europe), or http://www.speedcamupdates.com (Worldwide) and update that offlinespeedcams.dat file ourselves!
It is worth mentioning that the "Sygic way" of updating these cameras is to create a .rupi file and use this file as your own speedcam/poi database. But I believe it is best to modify/rebuild the build-in .dat file, as it offers some "improvements" over the .rupi update method.
Requirements:
- Python 2.7
- The .py script
The script below will convert iGO's .txt speedcam format to Sygic's .dat. So in order to create a Sygic offline build-in database you'll have to grab an iGO database as source, which is pretty common.
Convert:
python offlinespeedcams.py thefile.txt
The command above will create (or update if a file called offlinespeedcams.dat is present) a speedcam offline database file that you can upload to your Sygic GPS.
Update Sygic App
On Android the path is:
INTERNAL_MEMORY/Android/data/com.sygic.aura/files/Maps/speedcams
Once you've created/updated your offlinespeedcams.py file, upload it to that directory and restart the Sygic app, and you'll be ready to go!
The script will also create a .html that will display a map with all cameras loaded into the .dat file.
Download the python script offlinespeedcams.py
The Script
# -*- coding: utf-8 -*-
from __future__ import print_function, absolute_import, unicode_literals
import csv
import sqlite3
import os
import re
import datetime
def igo2sygic(files, igo_types, debug):
speedcams = [] # [[latitude, longitude, speed, type], [...]]
for filename in files:
if not os.path.exists(filename):
continue
if debug:
print('\n' + filename)
with open(filename, 'rb') as csvfile:
speedcam_csv = csv.reader(csvfile, delimiter=b',', quotechar=b'"')
for row in speedcam_csv:
#omit header
if speedcam_csv.line_num == 1 and row[0] == 'X':
continue
#omit bad lines
if len(row) != 6:
if debug:
print(speedcam_csv.line_num, 'BAD LINE !!!', sep='; ')
continue
longitude, latitude, kind, speed, dirtype, angle = row[0], row[1], row[2], row[3], row[4], row[5]
#is latitude a float? if not, omit record
try:
float(latitude)
except ValueError:
if debug:
print(speedcam_csv.line_num, 'Y', latitude, sep='; ')
continue
#is longitude a float? if not, omit record
try:
float(longitude)
except ValueError:
if debug:
print(speedcam_csv.line_num, 'X', longitude, sep='; ')
continue
#is speed a integer? if not, try to find or set zero
if not speed.isdigit():
#try to find first number in string
speed = re.search('\d+|$', speed).group()
if not speed.isdigit():
speed = 0
if debug:
print(speedcam_csv.line_num, 'SPEED', row[3], speed, sep='; ')
if igo_types:
#is type a integer? if not, set as normal speed camera - 1
if not kind.isdigit():
kind = '1'
if debug:
print(speedcam_csv.line_num, 'TYPE', row[2], kind, sep='; ')
if kind in igo_types:
kind = igo_types[kind]
else:
if debug and igo_types:
print(speedcam_csv.line_num, 'TYPE', row[2], 'not defined in igotypes', sep='; ')
continue
else:
kind = '1'
#igo dirtype equals 0 or 2: both ways; dirtype equals 1: single direction
both_ways = 0 if int(dirtype) == 1 else 1
#convert to sygic format
speedcams.append([int(float(latitude) * 100000), int(float(longitude) * 100000), int(speed), int(kind), int(angle), int(both_ways)])
#sort by latitude, longitude, speed
speedcams.sort(key=lambda x: (x[0], x[1], x[2]))
print('\nSpeedCameras all: {:,}'.format(len(speedcams)))
#eliminate duplicates:
#append duplicated speed cameras to radars dict group by location
radars = {} #radars[location: latitude,longitude] = [[index, latitude, longitude, speed, mark_to_del: 0 - ok; 1 - del], [...], [...]]
INDEX = 0
SPEED = 3
MARK = 4
speedcams2 = speedcams[:]
index1 = len(speedcams2)
while speedcams2:
index1 -= 1
latitude1, longitude1, speed1, kind1, angle1, both_ways1 = speedcams2.pop()
location = str(latitude1) + ',' + str(longitude1)
if location in radars:
continue
index2 = index1
for speedcam2 in reversed(speedcams2):
index2 -= 1
latitude2, longitude2, speed2, kind2, angle2, both_ways2 = speedcam2
if latitude1 == latitude2 and longitude1 == longitude2:
if location not in radars:
radars[location] = [[index1, latitude1, longitude1, speed1, kind1, angle1, both_ways1]]
radars[location].append([index2, latitude2, longitude2, speed2, kind2, angle2, both_ways2])
else:
break
#if exactly the same: leave first, mark rest to delete
for duplicates in radars.values():
for idx1 in range(len(duplicates)):
for idx2, radar in enumerate(duplicates):
if idx2 > idx1 and radar[SPEED] == duplicates[idx1][SPEED]:
radar[MARK] = 1
def count2leave(duplicates):
return sum(1 for radar in duplicates if radar[MARK] == 0)
#one of speed is zero
for duplicates in radars.values():
if count2leave(duplicates) > 1:
for radar in duplicates:
if radar[SPEED] == 0:
radar[MARK] = 1
#select lowest speed limit
for duplicates in radars.values():
if count2leave(duplicates) > 1:
minspeed = min([radar[SPEED] for radar in duplicates if radar[MARK] == 0])
for radar in duplicates:
if radar[MARK] == 0 and radar[SPEED] != minspeed:
radar[MARK] = 1
if debug:
for location, duplicates in radars.items():
print('\n', location, count2leave(duplicates))
for radar in duplicates:
print(radar)
#delete marked
del_list = [radar[INDEX] for duplicates in radars.values() for radar in duplicates if radar[MARK] == 1]
del_list.sort(key=int, reverse=True)
[speedcams.pop(d) for d in del_list]
return speedcams
def dat2points(dat_filename):
if not os.path.exists(dat_filename):
return
conn = sqlite3.connect(dat_filename, isolation_level=None)
cursor = conn.cursor()
cursor.execute('SELECT Latitude, Longitude, SpeedLimit, Type, Angle, BothWays FROM OfflineSpeedcam ORDER BY Latitude, Longitude')
return cursor.fetchall()
def points2map(speedcams):
html_filename = 'offlinespeedcams.dat_' + datetime.datetime.now().strftime('%Y%m%d_%H%M%S')
html = []
html.append('<!DOCTYPE html>')
html.append('<html>')
html.append('<head>')
html.append(' <title>' + html_filename + '</title>')
html.append(' <meta charset="utf-8">')
html.append(' <style>')
html.append(' #map {')
html.append(' height: 100%;')
html.append(' }')
html.append('')
html.append(' html, body {')
html.append(' height: 100%;')
html.append(' margin: 0;')
html.append(' padding: 0;')
html.append(' }')
html.append(' </style>')
html.append('</head>')
html.append('<body>')
html.append('<div id="map"></div>')
html.append('<script>')
html.append('')
html.append(' var sygicTypes = {')
html.append(' 0: {name: "RADAR_SYMBOL", symbol: "UE37A"},')
html.append(' 1: {name: "RADAR_STATIC_SPEED", symbol: "UE37B"},')
html.append(' 2: {name: "RADAR_STATIC_RED_LIGHT", symbol: "UE37E"},')
html.append(' 3: {name: "RADAR_SEMIMOBILE_SPEED", symbol: "UE37B"},')
html.append(' 4: {name: "RADAR_STATIC_AVERAGE_SPEED", symbol: "UE37C"},')
html.append(' 5: {name: "RADAR_MOBILE_SPEED", symbol: "UE37B"},')
html.append(' 6: {name: "RADAR_STATIC_RED_LIGHT_SPEED", symbol: "UE37E"},')
html.append(' 7: {name: "RADAR_MOBILE_RED_LIGHT", symbol: "UE37E"},')
html.append(' 8: {name: "RADAR_MOBILE_AVERAGE_SPEED", symbol: "UE37C"},')
html.append(' 9: {name: "RADAR_FAV_COPS_PLACE", symbol: "UE37D"},')
html.append(' 10: {name: "RADAR_INFO_CAMERA", symbol: "UE37A"},')
html.append(' 11: {name: "RADAR_DANGEROUS_PLACE", symbol: "UE37A"},')
html.append(' 12: {name: "RADAR_CONGESTION", symbol: "UE37F"},')
html.append(' 13: {name: "RADAR_WEIGHT_CHECK", symbol: "UE37A"},')
html.append(' 14: {name: "RADAR_DISTANCE_CHECK", symbol: "UE37A"},')
html.append(' 15: {name: "RADAR_CLOSURE", symbol: "UE030"},')
html.append(' 16: {name: "RADAR_SCHOOLZONE", symbol: "UE02E"}')
html.append(' };')
html.append('')
html.append('')
html.append(
' var UE37A = "";')
html.append(
' var UE37B = "";')
html.append(
' var UE37C = "";')
html.append(
' var UE37D = "";')
html.append(
' var UE37E = "";')
html.append(
' var UE37F = "";')
html.append(
' var UE02E = "";')
html.append(' var UE030 = "";')
html.append('')
html.append(' var map;')
html.append(' var markers = [];')
html.append(' var types = {};')
html.append('')
html.append(' function initMap() {')
html.append(' map = new google.maps.Map(document.getElementById("map"), {')
html.append(' zoom: 5,')
html.append(' center: new google.maps.LatLng(48.208775, 16.372477)')
html.append(' });')
html.append('')
html.append(' var speedCams = [')
for sc in speedcams:
latitude, longitude, speed_limit, kind, angle, both_ways = sc
latitude = str(float(latitude) / 100000)
longitude = str(float(longitude) / 100000)
html.append('{y:' + latitude + ',x:' + longitude + ',s:' + str(speed_limit) + ',t:' + str(kind) + '},')
html.append(' ];')
html.append('')
html.append(' speedCams.forEach(function (speedCam) {')
html.append('')
html.append(' types[speedCam.t] = (speedCam.t in types) ? types[speedCam.t] += 1 : 1;')
html.append('')
html.append(' markers.push(')
html.append(' new google.maps.Marker({')
html.append(' map: map,')
html.append(' position: new google.maps.LatLng(speedCam.y, speedCam.x),')
html.append(' title: "SPEED: " + speedCam.t + "; TYPE: " + speedCam.t + ";",')
html.append(' type: speedCam.t,')
html.append(' visible: false')
html.append(' }));')
html.append(' });')
html.append('')
html.append(' function markersShowHide(type, visible) {')
html.append(' console.log(type, visible);')
html.append(' markers.forEach(function (marker) {')
html.append(' if (marker.type == type) {')
html.append(' marker.setVisible(visible);')
html.append(' }')
html.append(' });')
html.append(' }')
html.append('')
html.append(' var controlUI = document.createElement("div");')
html.append(' controlUI.style.backgroundColor = "#fff";')
html.append(' controlUI.style.border = "2px solid #fff";')
html.append(' controlUI.style.borderRadius = "3px";')
html.append(' controlUI.style.boxShadow = "0 2px 6px rgba(0,0,0,.3)";')
html.append(' controlUI.style.lineHeight = "35px";')
html.append(' controlUI.style.marginLeft = "10px";')
html.append(' controlUI.style.textAlign = "left";')
html.append(' controlUI.title = "Filter markers";')
html.append('')
html.append(' var centerControlDiv = document.createElement("div");')
html.append(' centerControlDiv.index = 1;')
html.append(' centerControlDiv.appendChild(controlUI);')
html.append('')
html.append(' for (var type in types) {')
html.append(' var controlCheckBox = document.createElement("input");')
html.append(' controlCheckBox.value = type;')
html.append(' controlCheckBox.type = "checkbox";')
html.append(' controlCheckBox.checked = false;')
html.append(' controlCheckBox.style.margin = "0px 5px 0px 5px";')
html.append(' controlCheckBox.style.verticalAlign = "middle";')
html.append('')
html.append(' controlCheckBox.addEventListener("click", function () {')
html.append(' markersShowHide(this.value, this.checked);')
html.append(' });')
html.append('')
html.append(' var controlImage = document.createElement("img");')
html.append(' controlImage.src = (type in sygicTypes) ? eval(sygicTypes[type].symbol) : UE37A;')
html.append(' controlImage.width = "24";')
html.append(' controlImage.height = "24";')
html.append(' controlImage.style.backgroundColor = "rgba(0,0,0,0.7)";')
html.append(' controlImage.style.borderRadius = "10%";')
html.append(' controlImage.style.margin = "0px 5px 0px 5px";')
html.append(' controlImage.style.verticalAlign = "middle";')
html.append('')
html.append(' var controlText = document.createElement("span");')
html.append(' controlText.style.color = "rgb(25,25,25)";')
html.append(' controlText.style.fontFamily = "Roboto,Arial,sans-serif";')
html.append(' controlText.style.fontSize = "11px";')
html.append(' controlText.style.lineHeight = "38px";')
html.append(' controlText.style.verticalAlign = "middle";')
html.append(' controlText.innerHTML = type.toString() + " " + ((type in sygicTypes) ? sygicTypes[type].name : "") + " (" + types[type] + ")";')
html.append('')
html.append(' var controlHolder = document.createElement("div");')
html.append(' controlHolder.style.paddingLeft = "5px";')
html.append(' controlHolder.style.paddingRight = "5px";')
html.append(' controlHolder.style.whiteSpace = "nowrap";')
html.append('')
html.append(' controlHolder.appendChild(controlCheckBox);')
html.append(' controlHolder.appendChild(controlImage);')
html.append(' controlHolder.appendChild(controlText);')
html.append('')
html.append(' controlUI.appendChild(controlHolder);')
html.append(' }')
html.append('')
html.append(' map.controls[google.maps.ControlPosition.LEFT_CENTER].push(centerControlDiv);')
html.append(' }')
html.append('</script>')
html.append('<script async defer src="https://maps.googleapis.com/maps/api/js?callback=initMap"></script>')
html.append('</body>')
html.append('</html>')
with open(html_filename + '.html', 'w') as html_file:
html_file.write('\n'.join(html))
def save_dat(speedcams, dat_filename, unit, debug):
db_is_new = not os.path.exists(dat_filename)
speed_limit_units = 1 if unit == 'mph' else 0
conn = sqlite3.connect(dat_filename, isolation_level=None)
if db_is_new:
db_schema = """
CREATE TABLE Info (Version REAL not null, CreatedAt text not null, Note nvarchar(255) null);
CREATE TABLE OfflineSpeedcam (Id int not null, Latitude int not null, Longitude int not null, Type byte not null, Angle int null, BothWays bit, SpeedLimit int, Osm bit, PairId int null, SpeedLimitUnits byte null);
CREATE TABLE OfflineZone (Id int not null, Type byte not null, SpeedLimit smallint not null, LatitudeMin int not null, LongitudeMin int not null, LatitudeMax int not null, LongitudeMax int not null);
CREATE INDEX speedcamsLatLon ON OfflineSpeedcam (Latitude, Longitude);
"""
conn.executescript(db_schema)
conn.commit()
cursor = conn.cursor()
#get max id
cursor.execute('SELECT coalesce(max(Id), 0) AS max_id FROM OfflineSpeedcam')
max_id = cursor.fetchone()[0]
cursor.execute('SELECT Latitude, Longitude FROM OfflineSpeedcam ORDER BY Id')
offspeedcams = cursor.fetchall()
offspeedcams = set(offspeedcams)
#generate SQL
speedcams_added = []
sql = []
for sc in speedcams:
latitude, longitude, speed_limit, kind, angle, both_ways = sc
if (latitude, longitude) not in offspeedcams:
max_id += 1
sql.append('INSERT INTO OfflineSpeedcam (Id, Latitude, Longitude, Type, Angle, BothWays, SpeedLimit, Osm, PairId, SpeedLimitUnits) VALUES ({Id}, {Latitude}, {Longitude}, {Type}, {Angle}, {BothWays}, {SpeedLimit}, 0, NULL, {SpeedLimitUnits});'.format(Id=max_id, Latitude=latitude, Longitude=longitude, SpeedLimit=speed_limit, Type=kind, Angle=angle, BothWays=both_ways, SpeedLimitUnits=speed_limit_units))
speedcams_added.append(sc)
else:
if debug:
print('Already exists in db', (latitude, longitude))
if sql:
sql.append('DELETE FROM Info;')
sql.append('INSERT INTO Info (Version, CreatedAt, Note) VALUES ({version}, "{createdAt}", NULL);'.format(version=2, createdAt=datetime.datetime.now().strftime('%Y-%m-%d %H:%M:%S')))
conn.executescript('BEGIN TRANSACTION')
conn.executescript(''.join(sql))
conn.executescript('COMMIT')
return speedcams_added
if __name__ == '__main__':
import argparse
import fnmatch
import locale
arg_parser = argparse.ArgumentParser(description='Sygic offlinespeedcams.dat generator. Convert Speed Camera / Photo Radar from IGO to Sygic', formatter_class=argparse.ArgumentDefaultsHelpFormatter)
arg_parser.add_argument('-t', '--type', choices=['igo'], default='igo', help='Input type')
arg_parser.add_argument('-d', '--dat', type=str, default='offlinespeedcams.dat', help='DAT file')
arg_parser.add_argument('-u', '--unit', choices=['kmh','mph'], default='kmh', help='Unit: kmh or mph')
arg_parser.add_argument('-it', '--igotypes', action=type(b'', (argparse.Action,), dict(__call__=lambda self, parser, namespace, values, option_string: getattr(namespace, self.dest).update(dict([v.split('=') for v in values.replace(';', ',').split(',') if len(v.split('=')) == 2])))), default={'1':'1','2':'6','3':'2','4':'4','5':'5','6':'2','7':'2','8':'11','9':'16','10':'10','11':'6','12':'2','13':'10','15':'12','17':'9','31':'11'}, metavar='KEY1=VAL1,KEY2=VAL2;KEY3=VAL3...', dest='igo_types', help='You can specific your own types, first IGO, second Sygic')
arg_parser.add_argument('--debug', action='store_true', help='Print debug data')
arg_parser.add_argument('--map', action='store_true', default=True, help='Generate Google Maps with added points')
arg_parser.add_argument('--dat2map', action='store_true', default=False, help='Generate Google Maps with points from offlinespeedcams.dat')
arg_parser.add_argument('files', nargs='*', help='Source files')
args = arg_parser.parse_args()
def list_dir(cur_dir, mask, files_list):
cur_dir = os.path.normpath(cur_dir)
for f in os.listdir(cur_dir):
file_name = os.path.normpath(os.path.join(cur_dir, f))
if not f.startswith('.') and os.path.isdir(file_name):
list_dir(file_name, mask, files_list)
if f.startswith('.'):
continue
if fnmatch.fnmatch(f, mask) and os.path.isfile(file_name):
files_list.append(file_name)
all_files = []
for filename in args.files:
filename = os.path.abspath(os.path.normpath(unicode(filename, locale.getpreferredencoding())))
if os.path.isfile(filename):
all_files.append(filename)
else:
if os.path.isdir(filename):
list_dir(os.path.dirname(filename), '*.*', all_files)
else:
list_dir(os.path.dirname(filename), os.path.basename(filename), all_files)
if args.type == 'igo' and all_files:
speedcams = igo2sygic(all_files, args.igo_types, args.debug)
print('\nSpeedCameras after cleaning: {:,}'.format(len(speedcams)))
speedcams_added = save_dat(speedcams, args.dat, args.unit, args.debug)
print('\nSpeedCameras added: {:,}'.format(len(speedcams_added)))
if args.map and speedcams_added:
points2map(speedcams_added)
if args.dat2map:
dat_points = dat2points(args.dat)
if dat_points:
points2map(dat_points)
{{ 'Comments (%count%)' | trans {count:count} }}
{{ 'Comments are closed.' | trans }}