Nominatim/test/python/tools/test_tiger_data.py

168 lines
6.0 KiB
Python
Raw Normal View History

# SPDX-License-Identifier: GPL-3.0-or-later
2022-01-03 18:23:58 +03:00
#
# This file is part of Nominatim. (https://nominatim.org)
#
# Copyright (C) 2024 by the Nominatim developer community.
2022-01-03 18:23:58 +03:00
# For a full list of authors see the git log.
2021-03-10 14:48:12 +03:00
"""
Test for tiger data function
"""
2021-05-20 00:07:39 +03:00
import tarfile
2021-05-14 00:37:51 +03:00
from textwrap import dedent
2021-03-10 14:48:12 +03:00
import pytest
2024-07-05 11:43:10 +03:00
import pytest_asyncio
2021-03-10 14:48:12 +03:00
from nominatim_db.db.connection import execute_scalar
from nominatim_db.tools import tiger_data, freeze
2024-06-27 22:26:12 +03:00
from nominatim_db.errors import UsageError
2021-03-10 14:48:12 +03:00
2021-05-14 00:37:51 +03:00
class MockTigerTable:
def __init__(self, conn):
self.conn = conn
with conn.cursor() as cur:
cur.execute("""CREATE TABLE tiger (linegeo GEOMETRY,
start INTEGER,
stop INTEGER,
interpol TEXT,
token_info JSONB,
postcode TEXT)""")
# We need this table to determine if the database is frozen or not
cur.execute("CREATE TABLE place (number INTEGER)")
2021-05-14 00:37:51 +03:00
def count(self):
return execute_scalar(self.conn, "SELECT count(*) FROM tiger")
2021-05-14 00:37:51 +03:00
def row(self):
with self.conn.cursor() as cur:
cur.execute("SELECT * FROM tiger LIMIT 1")
return cur.fetchone()
@pytest.fixture
def tiger_table(def_config, temp_db_conn, sql_preprocessor,
temp_db_with_extensions, tmp_path):
def_config.lib_dir.sql = tmp_path / 'sql'
def_config.lib_dir.sql.mkdir()
(def_config.lib_dir.sql / 'tiger_import_start.sql').write_text(
"""CREATE OR REPLACE FUNCTION tiger_line_import(linegeo GEOMETRY, start INTEGER,
stop INTEGER, interpol TEXT,
token_info JSONB, postcode TEXT)
RETURNS INTEGER AS $$
2021-05-20 00:07:39 +03:00
INSERT INTO tiger VALUES(linegeo, start, stop, interpol, token_info, postcode)
RETURNING 1
2021-05-14 00:37:51 +03:00
$$ LANGUAGE SQL;""")
(def_config.lib_dir.sql / 'tiger_import_finish.sql').write_text(
"""DROP FUNCTION tiger_line_import (linegeo GEOMETRY, in_startnumber INTEGER,
in_endnumber INTEGER, interpolationtype TEXT,
token_info JSONB, in_postcode TEXT);""")
return MockTigerTable(temp_db_conn)
2021-03-10 14:48:12 +03:00
2021-05-14 00:37:51 +03:00
@pytest.fixture
def csv_factory(tmp_path):
def _mk_file(fname, hnr_from=1, hnr_to=9, interpol='odd', street='Main St',
city='Newtown', state='AL', postcode='12345',
geometry='LINESTRING(-86.466995 32.428956,-86.466923 32.428933)'):
(tmp_path / (fname + '.csv')).write_text(dedent("""\
from;to;interpolation;street;city;state;postcode;geometry
{};{};{};{};{};{};{};{}
""".format(hnr_from, hnr_to, interpol, street, city, state,
postcode, geometry)))
return _mk_file
2021-03-10 15:44:17 +03:00
@pytest.mark.parametrize("threads", (1, 5))
2024-07-05 11:43:10 +03:00
@pytest.mark.asyncio
async def test_add_tiger_data(def_config, src_dir, tiger_table, tokenizer_mock, threads):
await tiger_data.add_tiger_data(str(src_dir / 'test' / 'testdb' / 'tiger'),
def_config, threads, tokenizer_mock())
2021-05-14 00:37:51 +03:00
assert tiger_table.count() == 6213
2024-07-05 11:43:10 +03:00
@pytest.mark.asyncio
async def test_add_tiger_data_database_frozen(def_config, temp_db_conn, tiger_table, tokenizer_mock,
tmp_path):
freeze.drop_update_tables(temp_db_conn)
with pytest.raises(UsageError) as excinfo:
2024-07-05 11:43:10 +03:00
await tiger_data.add_tiger_data(str(tmp_path), def_config, 1, tokenizer_mock())
assert "database frozen" in str(excinfo.value)
assert tiger_table.count() == 0
2024-07-05 11:43:10 +03:00
@pytest.mark.asyncio
async def test_add_tiger_data_no_files(def_config, tiger_table, tokenizer_mock,
2021-05-14 00:37:51 +03:00
tmp_path):
2024-07-05 11:43:10 +03:00
await tiger_data.add_tiger_data(str(tmp_path), def_config, 1, tokenizer_mock())
2021-05-14 00:37:51 +03:00
assert tiger_table.count() == 0
2024-07-05 11:43:10 +03:00
@pytest.mark.asyncio
async def test_add_tiger_data_bad_file(def_config, tiger_table, tokenizer_mock,
2021-05-14 00:37:51 +03:00
tmp_path):
sqlfile = tmp_path / '1010.csv'
sqlfile.write_text("""Random text""")
2024-07-05 11:43:10 +03:00
await tiger_data.add_tiger_data(str(tmp_path), def_config, 1, tokenizer_mock())
2021-05-14 00:37:51 +03:00
assert tiger_table.count() == 0
2024-07-05 11:43:10 +03:00
@pytest.mark.asyncio
async def test_add_tiger_data_hnr_nan(def_config, tiger_table, tokenizer_mock,
2021-05-14 00:37:51 +03:00
csv_factory, tmp_path):
csv_factory('file1', hnr_from=99)
csv_factory('file2', hnr_from='L12')
csv_factory('file3', hnr_to='12.4')
2024-07-05 11:43:10 +03:00
await tiger_data.add_tiger_data(str(tmp_path), def_config, 1, tokenizer_mock())
2021-05-14 00:37:51 +03:00
assert tiger_table.count() == 1
2024-07-05 11:43:10 +03:00
assert tiger_table.row().start == 99
2021-03-10 15:44:17 +03:00
@pytest.mark.parametrize("threads", (1, 5))
2024-07-05 11:43:10 +03:00
@pytest.mark.asyncio
async def test_add_tiger_data_tarfile(def_config, tiger_table, tokenizer_mock,
2021-05-14 00:37:51 +03:00
tmp_path, src_dir, threads):
2021-04-19 13:23:01 +03:00
tar = tarfile.open(str(tmp_path / 'sample.tar.gz'), "w:gz")
2021-05-14 00:37:51 +03:00
tar.add(str(src_dir / 'test' / 'testdb' / 'tiger' / '01001.csv'))
tar.close()
2024-07-05 11:43:10 +03:00
await tiger_data.add_tiger_data(str(tmp_path / 'sample.tar.gz'), def_config, threads,
tokenizer_mock())
2021-05-14 00:37:51 +03:00
assert tiger_table.count() == 6213
2021-05-14 00:37:51 +03:00
2024-07-05 11:43:10 +03:00
@pytest.mark.asyncio
async def test_add_tiger_data_bad_tarfile(def_config, tiger_table, tokenizer_mock,
2021-05-14 00:37:51 +03:00
tmp_path):
tarfile = tmp_path / 'sample.tar.gz'
tarfile.write_text("""Random text""")
with pytest.raises(UsageError):
2024-07-05 11:43:10 +03:00
await tiger_data.add_tiger_data(str(tarfile), def_config, 1, tokenizer_mock())
2021-05-14 00:37:51 +03:00
2024-07-05 11:43:10 +03:00
@pytest.mark.asyncio
async def test_add_tiger_data_empty_tarfile(def_config, tiger_table, tokenizer_mock,
2021-05-20 00:07:39 +03:00
tmp_path):
2021-04-19 13:23:01 +03:00
tar = tarfile.open(str(tmp_path / 'sample.tar.gz'), "w:gz")
2021-05-14 00:37:51 +03:00
tar.add(__file__)
2021-03-10 15:44:17 +03:00
tar.close()
2024-07-05 11:43:10 +03:00
await tiger_data.add_tiger_data(str(tmp_path / 'sample.tar.gz'), def_config, 1,
tokenizer_mock())
2021-05-14 00:37:51 +03:00
assert tiger_table.count() == 0