]> git.openstreetmap.org Git - nominatim.git/blob - test/python/db/test_async_connection.py
Merge remote-tracking branch 'upstream/master'
[nominatim.git] / test / python / db / test_async_connection.py
1 # SPDX-License-Identifier: GPL-2.0-only
2 #
3 # This file is part of Nominatim. (https://nominatim.org)
4 #
5 # Copyright (C) 2022 by the Nominatim developer community.
6 # For a full list of authors see the git log.
7 """
8 Tests for function providing a non-blocking query interface towards PostgreSQL.
9 """
10 from contextlib import closing
11 import concurrent.futures
12
13 import pytest
14 import psycopg2
15
16 from nominatim.db.async_connection import DBConnection, DeadlockHandler
17
18
19 @pytest.fixture
20 def conn(temp_db):
21     with closing(DBConnection('dbname=' + temp_db)) as connection:
22         yield connection
23
24
25 @pytest.fixture
26 def simple_conns(temp_db):
27     conn1 = psycopg2.connect('dbname=' + temp_db)
28     conn2 = psycopg2.connect('dbname=' + temp_db)
29
30     yield conn1.cursor(), conn2.cursor()
31
32     conn1.close()
33     conn2.close()
34
35
36 def test_simple_query(conn, temp_db_conn):
37     conn.connect()
38
39     conn.perform('CREATE TABLE foo (id INT)')
40     conn.wait()
41
42     temp_db_conn.table_exists('foo')
43
44
45 def test_wait_for_query(conn):
46     conn.connect()
47
48     conn.perform('SELECT pg_sleep(1)')
49
50     assert not conn.is_done()
51
52     conn.wait()
53
54
55 def test_bad_query(conn):
56     conn.connect()
57
58     conn.perform('SELECT efasfjsea')
59
60     with pytest.raises(psycopg2.ProgrammingError):
61         conn.wait()
62
63
64 def test_bad_query_ignore(temp_db):
65     with closing(DBConnection('dbname=' + temp_db, ignore_sql_errors=True)) as conn:
66         conn.connect()
67
68         conn.perform('SELECT efasfjsea')
69
70         conn.wait()
71
72
73 def exec_with_deadlock(cur, sql, detector):
74     with DeadlockHandler(lambda *args: detector.append(1)):
75         cur.execute(sql)
76
77
78 def test_deadlock(simple_conns):
79     cur1, cur2 = simple_conns
80
81     cur1.execute("""CREATE TABLE t1 (id INT PRIMARY KEY, t TEXT);
82                     INSERT into t1 VALUES (1, 'a'), (2, 'b')""")
83     cur1.connection.commit()
84
85     cur1.execute("UPDATE t1 SET t = 'x' WHERE id = 1")
86     cur2.execute("UPDATE t1 SET t = 'x' WHERE id = 2")
87
88     # This is the tricky part of the test. The first SQL command runs into
89     # a lock and blocks, so we have to run it in a separate thread. When the
90     # second deadlocking SQL statement is issued, Postgresql will abort one of
91     # the two transactions that cause the deadlock. There is no way to tell
92     # which one of the two. Therefore wrap both in a DeadlockHandler and
93     # expect that exactly one of the two triggers.
94     with concurrent.futures.ThreadPoolExecutor(max_workers=1) as executor:
95         deadlock_check = []
96         try:
97             future = executor.submit(exec_with_deadlock, cur2,
98                                      "UPDATE t1 SET t = 'y' WHERE id = 1",
99                                      deadlock_check)
100
101             while not future.running():
102                 pass
103
104
105             exec_with_deadlock(cur1, "UPDATE t1 SET t = 'y' WHERE id = 2",
106                                deadlock_check)
107         finally:
108             # Whatever happens, make sure the deadlock gets resolved.
109             cur1.connection.rollback()
110
111         future.result()
112
113         assert len(deadlock_check) == 1