diff --git a/scripts/borders.sql b/scripts/borders.sql
index c43bb74..00094e5 100644
--- a/scripts/borders.sql
+++ b/scripts/borders.sql
@@ -12,5 +12,16 @@ create table borders (
cmnt varchar(500)
);
+create table borders_backup (
+ backup varchar(30) not null,
+ name varchar(200) not null,
+ geom geometry not null,
+ disabled boolean not null default FALSE,
+ count_k integer,
+ modified timestamp not null,
+ cmnt varchar(500),
+ primary key (backup, name)
+);
+
create index border_idx on borders using gist (geom);
create index tiles_idx on tiles using gist (tile);
diff --git a/scripts/osm_borders.sh b/scripts/osm_borders.sh
index 72e8921..65258fe 100755
--- a/scripts/osm_borders.sh
+++ b/scripts/osm_borders.sh
@@ -65,7 +65,7 @@ fi
# 3. Make osm_borders table
echo Creating osm_borders table
-psql $DATABASE -c "drop table if exists osm_borders; create table osm_borders as select min(osm_id) as osm_id, ST_Transform(ST_Collect(way),4326) as way, admin_level::int as admin_level, coalesce(max(\"name:en\"), name) as name from planet_osm_polygon where boundary='administrative' and admin_level in ('2', '3', '4', '5', '6') group by name, admin_level;" || exit 3
+psql $DATABASE -c "drop table if exists osm_borders; create table osm_borders as select min(osm_id) as osm_id, ST_Buffer(ST_Transform(ST_Collect(way),4326), 0) as way, admin_level::int as admin_level, coalesce(max(\"name:en\"), name) as name from planet_osm_polygon where boundary='administrative' and admin_level in ('2', '3', '4', '5', '6') group by name, admin_level;" || exit 3
# 4. Copy it to the borders database
echo Copying osm_borders table to the borders database
diff --git a/server/borders-api.py b/server/borders-api.py
index eac4bf0..557218a 100755
--- a/server/borders-api.py
+++ b/server/borders-api.py
@@ -2,12 +2,13 @@
from flask import Flask, g, request, json, jsonify, abort, Response
from flask.ext.cors import CORS
from flask.ext.compress import Compress
-import psycopg2
from lxml import etree
from xml.sax.saxutils import quoteattr
+import psycopg2
TABLE = 'borders'
OSM_TABLE = 'osm_borders'
+BACKUP = 'borders_backup'
READONLY = False
app = Flask(__name__)
@@ -230,7 +231,13 @@ def divide():
if prefix != '':
prefix = '{}_'.format(prefix);
cur = g.conn.cursor()
- cur.execute('insert into {table} (geom, name, modified, count_k) select o.way as way, %s || name, now(), -1 from {osm}, (select way from {osm} where name like %s) r where ST_Contains(r.way, o.way) and {query};'.format(table=TABLE, osm=OSM_TABLE, query=query), (prefix, like,))
+ cur.execute('''insert into {table} (geom, name, modified, count_k)
+ select o.way as way, %s || name, now(), -1
+ from {osm} o, (
+ select way from {osm} where name like %s
+ ) r
+ where ST_Contains(r.way, o.way) and {query};
+ '''.format(table=TABLE, osm=OSM_TABLE, query=query), (prefix, like,))
cur.execute('delete from {} where name = %s;'.format(TABLE), (name,))
g.conn.commit()
return jsonify(status='ok')
@@ -273,6 +280,39 @@ def draw_hull():
g.conn.commit()
return jsonify(status='ok')
+@app.route('/backup')
+def backup_do():
+ cur = g.conn.cursor()
+ cur.execute("SELECT to_char(now(), 'IYYY-MM-DD HH24:MI'), max(backup) from {};".format(BACKUP))
+ (timestamp, tsmax) = cur.fetchone()
+ if timestamp == tsmax:
+ return jsonify(status='please try again later')
+ cur.execute('INSERT INTO {backup} (backup, name, geom, disabled, count_k, modified, cmnt) SELECT %s, name, geom, disabled, count_k, modified, cmnt from {table};'.format(backup=BACKUP, table=TABLE), (timestamp,))
+ g.conn.commit()
+ return jsonify(status='ok')
+
+@app.route('/restore')
+def backup_restore():
+ ts = request.args.get('timestamp')
+ cur = g.conn.cursor()
+ cur.execute('SELECT count(1) from {} where backup = %s;'.format(BACKUP), (ts,))
+ (count,) = cur.fetchone()
+ if count <= 0:
+ return jsonify(status='no such timestamp')
+ cur.execute('DELETE FROM {};'.format(TABLE))
+ cur.execute('INSERT INTO {table} (name, geom, disabled, count_k, modified, cmnt) SELECT name, geom, disabled, count_k, modified, cmnt from {backup} where backup = %s;'.format(backup=BACKUP, table=TABLE), (ts,))
+ g.conn.commit()
+ return jsonify(status='ok')
+
+@app.route('/backlist')
+def backup_list():
+ cur = g.conn.cursor()
+ cur.execute("SELECT backup, count(1) from {} group by backup order by backup desc;".format(BACKUP))
+ result = []
+ for res in cur:
+ result.append({ 'timestamp': res[0], 'text': res[0], 'count': res[1] })
+ return jsonify(backups=result)
+
@app.route('/josm')
def make_osm():
xmin = request.args.get('xmin')
@@ -496,7 +536,6 @@ def import_osm():
j = i + 1
while way[0] != way[-1] and j < len(multi):
print 'maybe way with start={}, end={}?'.format(multi[j]['nodes'][0], multi[j]['nodes'][-1])
- # todo: do not modify source way!!!
new_way = append_way(way, multi[j]['nodes'])
if new_way:
multi[i] = dict(multi[i])
@@ -553,7 +592,7 @@ def import_osm():
updated = updated + 1
else:
# create
- cur.execute('insert into {table} (name, disabled, geom, modified, count_k) values (%s, %s, %s, now(), -1);'.format(table=TABLE), (name, region['disabled'], region['wkt']))
+ cur.execute('insert into {table} (name, disabled, geom, modified, count_k) values (%s, %s, ST_GeomFromText(%s, 4326), now(), -1);'.format(table=TABLE), (name, region['disabled'], region['wkt']))
added = added + 1
g.conn.commit()
return jsonify(regions=len(regions), added=added, updated=updated)
diff --git a/www/borders.js b/www/borders.js
index 2ddf264..2a543f5 100644
--- a/www/borders.js
+++ b/www/borders.js
@@ -18,9 +18,9 @@ function init() {
map.addLayer(bordersLayer);
map.on('moveend', function() {
- if( map.getZoom() >= 4 )
+ if( map.getZoom() >= 5 )
updateBorders();
- $('#b_josm').css('visibility', map.getZoom() >= 8 ? 'visible' : 'hidden');
+ $('#b_josm').css('visibility', map.getZoom() >= 7 ? 'visible' : 'hidden');
});
document.getElementById('filefm').action = server + '/import';
@@ -147,7 +147,7 @@ function selectLayer(e) {
if( props['disabled'] )
e.target.setStyle({ fillOpacity: 0.01 });
$('#b_name').text(props['name']);
- $('#b_size').text(Math.round(props['count_k'] * 8 / 1000000) + ' MB');
+ $('#b_size').text(Math.round(props['count_k'] * 8 / 1024 / 1024) + ' MB');
//$('#b_nodes').text(borders[selectedId].layer.getLatLngs()[0].length);
$('#b_nodes').text(props['nodes']);
$('#b_date').text(props['modified']);
@@ -503,7 +503,7 @@ function bDivideDrawPreview(geojson) {
return;
divPreview = L.geoJson(geojson, {
style: function(f) {
- return { color: 'blue', weight: 1 };
+ return { color: 'blue', weight: 1, fill: false };
}
});
map.addLayer(divPreview);
@@ -551,3 +551,52 @@ function bHull() {
success: updateBorders
});
}
+
+function bBackup() {
+ $('#actions').css('display', 'none');
+ $('#backup_saving').css('display', 'none');
+ $('#backup_restoring').css('display', 'none');
+ $('#backup_save').attr('disabled', false);
+ $('#backup_list').text('');
+ $('#backup').css('display', 'block');
+ $.ajax(server + '/backlist', {
+ success: updateBackupList
+ });
+}
+
+function bBackupCancel() {
+ $('#actions').css('display', 'block');
+ $('#backup').css('display', 'none');
+}
+
+function updateBackupList(data) {
+ var list = $('#backup_list');
+ list.text('');
+ if( !data || !('backups' in data) )
+ return;
+ for( var i = 0; i < data.backups.length; i++ ) {
+ var b = data.backups[i];
+ var a = document.createElement('a');
+ a.href = '#';
+ a.onclick = (function(id, name) { return function() { bBackupRestore(id); return false } })(b['timestamp']);
+ list.append(a, $('
'));
+ $(a).text(b['text'] + ' (' + b['count'] + ')');
+ }
+}
+
+function bBackupSave() {
+ $.ajax(server + '/backup', {
+ success: bBackupCancel
+ });
+ $('#backup_save').attr('disabled', true);
+ $('#backup_saving').css('display', 'block');
+}
+
+function bBackupRestore(timestamp) {
+ $.ajax(server + '/restore', {
+ data: { 'timestamp': timestamp },
+ success: function() { bBackupCancel(); updateBorders(); }
+ });
+ $('#backup_list').text('');
+ $('#backup_restoring').css('display', 'block');
+}
diff --git a/www/index.html b/www/index.html
index 3638936..5cd0eaa 100644
--- a/www/index.html
+++ b/www/index.html
@@ -19,10 +19,11 @@
#osm_actions { display: none; margin-top: 1em; }
#info { margin-top: 2em; }
#b_delete, #b_clear { font-size: 8pt; }
- #rename, #split, #join, #point, #divide { display: none; }
+ #rename, #split, #join, #point, #divide, #backup { display: none; }
.actions input[type='text'] { width: 150px; }
#header { border-bottom: 1px solid gray; margin-bottom: 1em; padding-bottom: 1em; }
#f_topo, #f_chars, #f_comments { font-size: 10pt; }
+ #backup_saving, #backup_restoring { margin-bottom: 1em; }