3 from pathlib import Path
9 SRC_DIR = Path(__file__) / '..' / '..' / '..'
11 # always test against the source
12 sys.path.insert(0, str(SRC_DIR.resolve()))
14 from nominatim.config import Configuration
15 from nominatim.db import connection
17 class _TestingCursor(psycopg2.extras.DictCursor):
18 """ Extension to the DictCursor class that provides execution
19 short-cuts that simplify writing assertions.
22 def scalar(self, sql, params=None):
23 """ Execute a query with a single return value and return this value.
24 Raises an assertion when not exactly one row is returned.
26 self.execute(sql, params)
27 assert self.rowcount == 1
28 return self.fetchone()[0]
30 def row_set(self, sql, params=None):
31 """ Execute a query and return the result as a set of tuples.
33 self.execute(sql, params)
34 if self.rowcount == 1:
35 return set(tuple(self.fetchone()))
37 return set((tuple(row) for row in self))
39 def table_exists(self, table):
40 """ Check that a table with the given name exists in the database.
42 num = self.scalar("""SELECT count(*) FROM pg_tables
43 WHERE tablename = %s""", (table, ))
48 def temp_db(monkeypatch):
49 """ Create an empty database for the test. The database name is also
50 exported into NOMINATIM_DATABASE_DSN.
52 name = 'test_nominatim_python_unittest'
53 conn = psycopg2.connect(database='postgres')
55 conn.set_isolation_level(0)
56 with conn.cursor() as cur:
57 cur.execute('DROP DATABASE IF EXISTS {}'.format(name))
58 cur.execute('CREATE DATABASE {}'.format(name))
62 monkeypatch.setenv('NOMINATIM_DATABASE_DSN' , 'dbname=' + name)
66 conn = psycopg2.connect(database='postgres')
68 conn.set_isolation_level(0)
69 with conn.cursor() as cur:
70 cur.execute('DROP DATABASE IF EXISTS {}'.format(name))
77 return 'dbname=' + temp_db
81 def temp_db_with_extensions(temp_db):
82 conn = psycopg2.connect(database=temp_db)
83 with conn.cursor() as cur:
84 cur.execute('CREATE EXTENSION hstore; CREATE EXTENSION postgis;')
91 def temp_db_conn(temp_db):
92 """ Connection to the test database.
94 with connection.connect('dbname=' + temp_db) as conn:
99 def temp_db_cursor(temp_db):
100 """ Connection and cursor towards the test database. The connection will
101 be in auto-commit mode.
103 conn = psycopg2.connect('dbname=' + temp_db)
104 conn.set_isolation_level(0)
105 with conn.cursor(cursor_factory=_TestingCursor) as cur:
111 def table_factory(temp_db_cursor):
112 def mk_table(name, definition='id INT'):
113 temp_db_cursor.execute('CREATE TABLE {} ({})'.format(name, definition))
120 return Configuration(None, SRC_DIR.resolve() / 'settings')
124 return SRC_DIR.resolve()
127 def status_table(temp_db_conn):
128 """ Create an empty version of the status table and
129 the status logging table.
131 with temp_db_conn.cursor() as cur:
132 cur.execute("""CREATE TABLE import_status (
133 lastimportdate timestamp with time zone NOT NULL,
137 cur.execute("""CREATE TABLE import_osmosis_log (
145 temp_db_conn.commit()
149 def place_table(temp_db_with_extensions, temp_db_conn):
150 """ Create an empty version of the place table.
152 with temp_db_conn.cursor() as cur:
153 cur.execute("""CREATE TABLE place (
154 osm_id int8 NOT NULL,
155 osm_type char(1) NOT NULL,
159 admin_level smallint,
162 geometry Geometry(Geometry,4326) NOT NULL)""")
163 temp_db_conn.commit()
167 def place_row(place_table, temp_db_cursor):
168 """ A factory for rows in the place table. The table is created as a
169 prerequisite to the fixture.
171 idseq = itertools.count(1001)
172 def _insert(osm_type='N', osm_id=None, cls='amenity', typ='cafe', names=None,
173 admin_level=None, address=None, extratags=None, geom=None):
174 temp_db_cursor.execute("INSERT INTO place VALUES (%s, %s, %s, %s, %s, %s, %s, %s, %s)",
175 (osm_id or next(idseq), osm_type, cls, typ, names,
176 admin_level, address, extratags,
177 geom or 'SRID=4326;POINT(0 0 )'))
182 def placex_table(temp_db_with_extensions, temp_db_conn):
183 """ Create an empty version of the place table.
185 with temp_db_conn.cursor() as cur:
186 cur.execute("""CREATE TABLE placex (
187 place_id BIGINT NOT NULL,
188 parent_place_id BIGINT,
189 linked_place_id BIGINT,
191 indexed_date TIMESTAMP,
192 geometry_sector INTEGER,
193 rank_address SMALLINT,
194 rank_search SMALLINT,
196 indexed_status SMALLINT,
202 admin_level smallint,
205 geometry Geometry(Geometry,4326),
207 country_code varchar(2),
210 centroid GEOMETRY(Geometry, 4326))
212 temp_db_conn.commit()
216 def osm2pgsql_options(temp_db):
217 return dict(osm2pgsql='echo',
219 osm2pgsql_style='style.file',
221 dsn='dbname=' + temp_db,
223 tablespaces=dict(slim_data='', slim_index='',
224 main_data='', main_index=''))