2011-11-14 00:42:37 +00:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
2015-02-02 18:31:16 +00:00
|
|
|
from nose.tools import eq_
|
2013-06-03 02:28:24 +00:00
|
|
|
from utilities import execution_path, run_all
|
2011-11-14 00:42:37 +00:00
|
|
|
import threading
|
|
|
|
|
2011-11-23 11:33:58 +00:00
|
|
|
import os, mapnik
|
2011-11-14 00:42:37 +00:00
|
|
|
import sqlite3
|
|
|
|
|
|
|
|
def setup():
|
|
|
|
# All of the paths used are relative, if we run the tests
|
|
|
|
# from another directory we need to chdir()
|
|
|
|
os.chdir(execution_path('.'))
|
|
|
|
|
|
|
|
NUM_THREADS = 10
|
2011-11-14 01:00:34 +00:00
|
|
|
TOTAL = 245
|
2011-11-14 00:42:37 +00:00
|
|
|
|
2014-10-15 00:35:59 +00:00
|
|
|
def create_ds(test_db,table):
|
|
|
|
ds = mapnik.SQLite(file=test_db,table=table)
|
2015-02-02 18:31:16 +00:00
|
|
|
ds.all_features()
|
2014-10-15 00:49:04 +00:00
|
|
|
del ds
|
2011-11-14 00:42:37 +00:00
|
|
|
|
2012-09-05 11:53:37 +00:00
|
|
|
if 'sqlite' in mapnik.DatasourceCache.plugin_names():
|
2012-02-24 21:13:56 +00:00
|
|
|
|
2011-11-14 00:42:37 +00:00
|
|
|
def test_rtree_creation():
|
2014-10-15 00:35:59 +00:00
|
|
|
test_db = '../data/sqlite/world.sqlite'
|
|
|
|
index = test_db +'.index'
|
|
|
|
table = 'world_merc'
|
2011-11-14 00:42:37 +00:00
|
|
|
|
|
|
|
if os.path.exists(index):
|
|
|
|
os.unlink(index)
|
|
|
|
|
|
|
|
threads = []
|
|
|
|
for i in range(NUM_THREADS):
|
2014-10-15 00:35:59 +00:00
|
|
|
t = threading.Thread(target=create_ds,args=(test_db,table))
|
2011-11-14 00:42:37 +00:00
|
|
|
t.start()
|
|
|
|
threads.append(t)
|
|
|
|
|
|
|
|
for i in threads:
|
|
|
|
i.join()
|
2012-02-24 21:13:56 +00:00
|
|
|
|
2011-11-14 01:00:34 +00:00
|
|
|
eq_(os.path.exists(index),True)
|
|
|
|
conn = sqlite3.connect(index)
|
|
|
|
cur = conn.cursor()
|
2011-12-01 03:15:11 +00:00
|
|
|
try:
|
2014-10-15 00:35:59 +00:00
|
|
|
cur.execute("Select count(*) from idx_%s_GEOMETRY" % table.replace("'",""))
|
2011-12-01 03:15:11 +00:00
|
|
|
conn.commit()
|
|
|
|
eq_(cur.fetchone()[0],TOTAL)
|
|
|
|
except sqlite3.OperationalError:
|
2012-09-05 11:53:37 +00:00
|
|
|
# don't worry about testing # of index records if
|
2011-12-01 03:15:11 +00:00
|
|
|
# python's sqlite module does not support rtree
|
|
|
|
pass
|
2011-11-14 01:00:34 +00:00
|
|
|
cur.close()
|
2014-10-15 00:35:59 +00:00
|
|
|
conn.close()
|
2011-11-14 01:00:34 +00:00
|
|
|
|
2014-10-15 00:35:59 +00:00
|
|
|
ds = mapnik.SQLite(file=test_db,table=table)
|
2011-11-14 01:00:34 +00:00
|
|
|
fs = ds.all_features()
|
2014-10-15 00:49:04 +00:00
|
|
|
del ds
|
2011-11-14 01:00:34 +00:00
|
|
|
eq_(len(fs),TOTAL)
|
|
|
|
os.unlink(index)
|
2014-10-15 00:35:59 +00:00
|
|
|
ds = mapnik.SQLite(file=test_db,table=table,use_spatial_index=False)
|
2011-11-14 01:00:34 +00:00
|
|
|
fs = ds.all_features()
|
2014-10-15 00:49:04 +00:00
|
|
|
del ds
|
2011-11-14 01:00:34 +00:00
|
|
|
eq_(len(fs),TOTAL)
|
|
|
|
eq_(os.path.exists(index),False)
|
2011-11-14 00:42:37 +00:00
|
|
|
|
2014-10-15 00:35:59 +00:00
|
|
|
ds = mapnik.SQLite(file=test_db,table=table,use_spatial_index=True)
|
2011-11-19 19:39:59 +00:00
|
|
|
fs = ds.all_features()
|
2014-10-15 00:49:04 +00:00
|
|
|
#TODO - this loop is not releasing something
|
|
|
|
# because it causes the unlink below to fail on windows
|
|
|
|
# as the file is still open
|
|
|
|
#for feat in fs:
|
|
|
|
# query = mapnik.Query(feat.envelope())
|
|
|
|
# selected = ds.features(query)
|
|
|
|
# eq_(len(selected.features)>=1,True)
|
2014-10-15 00:35:59 +00:00
|
|
|
del ds
|
|
|
|
|
2011-11-19 19:39:59 +00:00
|
|
|
eq_(os.path.exists(index),True)
|
|
|
|
os.unlink(index)
|
|
|
|
|
2012-04-18 19:49:49 +00:00
|
|
|
def test_geometry_round_trip():
|
|
|
|
test_db = '/tmp/mapnik-sqlite-point.db'
|
2012-04-18 23:06:18 +00:00
|
|
|
ogr_metadata = True
|
2012-04-18 19:49:49 +00:00
|
|
|
|
|
|
|
# create test db
|
|
|
|
conn = sqlite3.connect(test_db)
|
|
|
|
cur = conn.cursor()
|
|
|
|
cur.execute('''
|
2012-04-18 23:06:18 +00:00
|
|
|
CREATE TABLE IF NOT EXISTS point_table
|
|
|
|
(id INTEGER PRIMARY KEY AUTOINCREMENT, geometry BLOB, name varchar)
|
2012-04-18 19:49:49 +00:00
|
|
|
''')
|
2012-04-18 23:06:18 +00:00
|
|
|
# optional: but nice if we want to read with ogr
|
|
|
|
if ogr_metadata:
|
|
|
|
cur.execute('''CREATE TABLE IF NOT EXISTS geometry_columns (
|
|
|
|
f_table_name VARCHAR,
|
|
|
|
f_geometry_column VARCHAR,
|
|
|
|
geometry_type INTEGER,
|
|
|
|
coord_dimension INTEGER,
|
|
|
|
srid INTEGER,
|
|
|
|
geometry_format VARCHAR )''')
|
|
|
|
cur.execute('''INSERT INTO geometry_columns
|
|
|
|
(f_table_name, f_geometry_column, geometry_format,
|
|
|
|
geometry_type, coord_dimension, srid) VALUES
|
|
|
|
('point_table','geometry','WKB', 1, 1, 4326)''')
|
2012-04-18 19:49:49 +00:00
|
|
|
conn.commit()
|
|
|
|
cur.close()
|
|
|
|
|
2012-04-18 23:06:18 +00:00
|
|
|
# add a point as wkb (using mapnik) to match how an ogr created db looks
|
|
|
|
x = -122 # longitude
|
|
|
|
y = 48 # latitude
|
|
|
|
wkt = 'POINT(%s %s)' % (x,y)
|
|
|
|
# little endian wkb (mapnik will auto-detect and ready either little or big endian (XDR))
|
|
|
|
wkb = mapnik.Path.from_wkt(wkt).to_wkb(mapnik.wkbByteOrder.NDR)
|
2012-04-18 19:49:49 +00:00
|
|
|
values = (None,sqlite3.Binary(wkb),"test point")
|
2012-04-18 23:06:18 +00:00
|
|
|
cur = conn.cursor()
|
2012-04-18 19:49:49 +00:00
|
|
|
cur.execute('''INSERT into "point_table" (id,geometry,name) values (?,?,?)''',values)
|
|
|
|
conn.commit()
|
|
|
|
cur.close()
|
2014-10-15 00:35:59 +00:00
|
|
|
conn.close()
|
2012-04-18 19:49:49 +00:00
|
|
|
|
2012-04-18 23:06:18 +00:00
|
|
|
def make_wkb_point(x,y):
|
|
|
|
import struct
|
|
|
|
byteorder = 1; # little endian
|
|
|
|
endianess = ''
|
|
|
|
if byteorder == 1:
|
|
|
|
endianess = '<'
|
|
|
|
else:
|
|
|
|
endianess = '>'
|
|
|
|
geom_type = 1; # for a point
|
|
|
|
return struct.pack('%sbldd' % endianess, byteorder, geom_type, x, y)
|
|
|
|
|
|
|
|
# confirm the wkb matches a manually formed wkb
|
|
|
|
wkb2 = make_wkb_point(x,y)
|
|
|
|
eq_(wkb,wkb2)
|
|
|
|
|
2012-04-18 19:49:49 +00:00
|
|
|
# ensure we can read this data back out properly with mapnik
|
|
|
|
ds = mapnik.Datasource(**{'type':'sqlite','file':test_db, 'table':'point_table'})
|
|
|
|
fs = ds.featureset()
|
|
|
|
feat = fs.next()
|
|
|
|
eq_(feat.id(),1)
|
|
|
|
eq_(feat['name'],'test point')
|
|
|
|
geoms = feat.geometries()
|
|
|
|
eq_(len(geoms),1)
|
2012-12-18 12:13:42 +00:00
|
|
|
eq_(geoms.to_wkt(),'Point(-122 48)')
|
2014-10-15 00:35:59 +00:00
|
|
|
del ds
|
2012-04-18 19:49:49 +00:00
|
|
|
|
|
|
|
# ensure it matches data read with just sqlite
|
2014-10-15 00:35:59 +00:00
|
|
|
conn = sqlite3.connect(test_db)
|
2012-04-18 19:49:49 +00:00
|
|
|
cur = conn.cursor()
|
|
|
|
cur.execute('''SELECT * from point_table''')
|
|
|
|
conn.commit()
|
|
|
|
result = cur.fetchone()
|
2012-04-18 23:06:18 +00:00
|
|
|
cur.close()
|
2012-04-18 19:49:49 +00:00
|
|
|
feat_id = result[0]
|
|
|
|
eq_(feat_id,1)
|
|
|
|
name = result[2]
|
|
|
|
eq_(name,'test point')
|
|
|
|
geom_wkb_blob = result[1]
|
2012-04-18 23:06:18 +00:00
|
|
|
eq_(str(geom_wkb_blob),geoms.to_wkb(mapnik.wkbByteOrder.NDR))
|
2012-04-18 19:49:49 +00:00
|
|
|
new_geom = mapnik.Path.from_wkb(str(geom_wkb_blob))
|
|
|
|
eq_(new_geom.to_wkt(),geoms.to_wkt())
|
2014-10-15 00:35:59 +00:00
|
|
|
conn.close()
|
2012-04-18 19:49:49 +00:00
|
|
|
os.unlink(test_db)
|
2011-11-19 19:39:59 +00:00
|
|
|
|
2011-11-14 00:42:37 +00:00
|
|
|
if __name__ == "__main__":
|
|
|
|
setup()
|
2014-10-15 00:35:59 +00:00
|
|
|
returncode = run_all(eval(x) for x in dir() if x.startswith("test_"))
|
|
|
|
exit(returncode)
|