Skip to content

Commit 9bfa2c4

Browse files
authored
Merge pull request #891 from ClearcodeHQ/clear-template
Have separate paratemeter for template and regular database name - cl…
2 parents d821171 + 6618341 commit 9bfa2c4

File tree

8 files changed

+60
-30
lines changed

8 files changed

+60
-30
lines changed

newsfragments/672.feature.rst

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
Have separate parameters for template database name and database name in DatabaseJanitor.
2+
It'll make it much clearer to understand the code and Janitor's behaviour.

pytest_postgresql/executor.py

+5
Original file line numberDiff line numberDiff line change
@@ -131,6 +131,11 @@ def __init__(
131131
},
132132
)
133133

134+
@property
135+
def template_dbname(self) -> str:
136+
"""Return the template database name."""
137+
return f"{self.dbname}_tmpl"
138+
134139
def start(self: T) -> T:
135140
"""Add check for postgresql version before starting process."""
136141
if self.version < self.MIN_SUPPORTED_VERSION:

pytest_postgresql/executor_noop.py

+5
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,11 @@ def __init__(
5656
self.dbname = dbname
5757
self._version: Any = None
5858

59+
@property
60+
def template_dbname(self) -> str:
61+
"""Return the template database name."""
62+
return f"{self.dbname}_tmpl"
63+
5964
@property
6065
def version(self) -> Any:
6166
"""Get postgresql's version."""

pytest_postgresql/factories/client.py

+8-2
Original file line numberDiff line numberDiff line change
@@ -75,9 +75,15 @@ def postgresql_factory(request: FixtureRequest) -> Iterator[Connection]:
7575
),
7676
category=DeprecationWarning,
7777
)
78-
7978
with DatabaseJanitor(
80-
pg_user, pg_host, pg_port, pg_db, proc_fixture.version, pg_password, isolation_level
79+
user=pg_user,
80+
host=pg_host,
81+
port=pg_port,
82+
dbname=pg_db,
83+
template_dbname=proc_fixture.template_dbname,
84+
version=proc_fixture.version,
85+
password=pg_password,
86+
isolation_level=isolation_level,
8187
) as janitor:
8288
db_connection: Connection = psycopg.connect(
8389
dbname=pg_db,

pytest_postgresql/factories/noprocess.py

+1-2
Original file line numberDiff line numberDiff line change
@@ -81,12 +81,11 @@ def postgresql_noproc_fixture(request: FixtureRequest) -> Iterator[NoopExecutor]
8181
dbname=pg_dbname,
8282
options=pg_options,
8383
)
84-
template_dbname = f"{noop_exec.dbname}_tmpl"
8584
with DatabaseJanitor(
8685
user=noop_exec.user,
8786
host=noop_exec.host,
8887
port=noop_exec.port,
89-
dbname=template_dbname,
88+
template_dbname=noop_exec.template_dbname,
9089
version=noop_exec.version,
9190
password=noop_exec.password,
9291
) as janitor:

pytest_postgresql/factories/process.py

+1-2
Original file line numberDiff line numberDiff line change
@@ -136,12 +136,11 @@ def postgresql_proc_fixture(
136136
# start server
137137
with postgresql_executor:
138138
postgresql_executor.wait_for_postgres()
139-
template_dbname = f"{postgresql_executor.dbname}_tmpl"
140139
with DatabaseJanitor(
141140
user=postgresql_executor.user,
142141
host=postgresql_executor.host,
143142
port=postgresql_executor.port,
144-
dbname=template_dbname,
143+
template_dbname=postgresql_executor.template_dbname,
145144
version=postgresql_executor.version,
146145
password=postgresql_executor.password,
147146
) as janitor:

pytest_postgresql/janitor.py

+24-21
Original file line numberDiff line numberDiff line change
@@ -26,8 +26,9 @@ def __init__(
2626
user: str,
2727
host: str,
2828
port: Union[str, int],
29-
dbname: str,
3029
version: Union[str, float, Version], # type: ignore[valid-type]
30+
dbname: Optional[str] = None,
31+
template_dbname: Optional[str] = None,
3132
password: Optional[str] = None,
3233
isolation_level: "Optional[psycopg.IsolationLevel]" = None,
3334
connection_timeout: int = 60,
@@ -38,6 +39,7 @@ def __init__(
3839
:param host: postgresql host
3940
:param port: postgresql port
4041
:param dbname: database name
42+
:param dbname: template database name
4143
:param version: postgresql version number
4244
:param password: optional postgresql password
4345
:param isolation_level: optional postgresql isolation level
@@ -49,7 +51,10 @@ def __init__(
4951
self.password = password
5052
self.host = host
5153
self.port = port
54+
# At least one of the dbname or template_dbname has to be filled.
55+
assert any([dbname, template_dbname])
5256
self.dbname = dbname
57+
self.template_dbname = template_dbname
5358
self._connection_timeout = connection_timeout
5459
self.isolation_level = isolation_level
5560
if not isinstance(version, Version):
@@ -59,36 +64,33 @@ def __init__(
5964

6065
def init(self) -> None:
6166
"""Create database in postgresql."""
62-
template_name = f"{self.dbname}_tmpl"
6367
with self.cursor() as cur:
64-
if self.dbname.endswith("_tmpl"):
65-
result = False
66-
else:
67-
cur.execute(
68-
"SELECT EXISTS "
69-
"(SELECT datname FROM pg_catalog.pg_database WHERE datname= %s);",
70-
(template_name,),
71-
)
72-
row = cur.fetchone()
73-
result = (row is not None) and row[0]
74-
if not result:
68+
if self.is_template():
69+
cur.execute(f'CREATE DATABASE "{self.template_dbname}";')
70+
elif self.template_dbname is None:
7571
cur.execute(f'CREATE DATABASE "{self.dbname}";')
7672
else:
7773
# All template database does not allow connection:
78-
self._dont_datallowconn(cur, template_name)
74+
self._dont_datallowconn(cur, self.template_dbname)
7975
# And make sure no-one is left connected to the template database.
80-
# Otherwise Creating database from template will fail
81-
self._terminate_connection(cur, template_name)
82-
cur.execute(f'CREATE DATABASE "{self.dbname}" TEMPLATE "{template_name}";')
76+
# Otherwise, Creating database from template will fail
77+
self._terminate_connection(cur, self.template_dbname)
78+
cur.execute(f'CREATE DATABASE "{self.dbname}" TEMPLATE "{self.template_dbname}";')
79+
80+
def is_template(self) -> bool:
81+
"""Determine whether the DatabaseJanitor maintains template or database."""
82+
return self.dbname is None
8383

8484
def drop(self) -> None:
8585
"""Drop database in postgresql."""
8686
# We cannot drop the database while there are connections to it, so we
8787
# terminate all connections first while not allowing new connections.
88+
db_to_drop = self.template_dbname if self.is_template() else self.dbname
89+
assert db_to_drop
8890
with self.cursor() as cur:
89-
self._dont_datallowconn(cur, self.dbname)
90-
self._terminate_connection(cur, self.dbname)
91-
cur.execute(f'DROP DATABASE IF EXISTS "{self.dbname}";')
91+
self._dont_datallowconn(cur, db_to_drop)
92+
self._terminate_connection(cur, db_to_drop)
93+
cur.execute(f'DROP DATABASE IF EXISTS "{db_to_drop}";')
9294

9395
@staticmethod
9496
def _dont_datallowconn(cur: Cursor, dbname: str) -> None:
@@ -113,12 +115,13 @@ def load(self, load: Union[Callable, str, Path]) -> None:
113115
* a callable that expects: host, port, user, dbname and password arguments.
114116
115117
"""
118+
db_to_load = self.template_dbname if self.is_template() else self.dbname
116119
_loader = build_loader(load)
117120
_loader(
118121
host=self.host,
119122
port=self.port,
120123
user=self.user,
121-
dbname=self.dbname,
124+
dbname=db_to_load,
122125
password=self.password,
123126
)
124127

tests/test_janitor.py

+14-3
Original file line numberDiff line numberDiff line change
@@ -15,14 +15,18 @@
1515
@pytest.mark.parametrize("version", (VERSION, 10, "10"))
1616
def test_version_cast(version: Any) -> None:
1717
"""Test that version is cast to Version object."""
18-
janitor = DatabaseJanitor("user", "host", "1234", "database_name", version)
18+
janitor = DatabaseJanitor(
19+
user="user", host="host", port="1234", dbname="database_name", version=version
20+
)
1921
assert janitor.version == VERSION
2022

2123

2224
@patch("pytest_postgresql.janitor.psycopg.connect")
2325
def test_cursor_selects_postgres_database(connect_mock: MagicMock) -> None:
2426
"""Test that the cursor requests the postgres database."""
25-
janitor = DatabaseJanitor("user", "host", "1234", "database_name", 10)
27+
janitor = DatabaseJanitor(
28+
user="user", host="host", port="1234", dbname="database_name", version=10
29+
)
2630
with janitor.cursor():
2731
connect_mock.assert_called_once_with(
2832
dbname="postgres", user="user", password=None, host="host", port="1234"
@@ -32,7 +36,14 @@ def test_cursor_selects_postgres_database(connect_mock: MagicMock) -> None:
3236
@patch("pytest_postgresql.janitor.psycopg.connect")
3337
def test_cursor_connects_with_password(connect_mock: MagicMock) -> None:
3438
"""Test that the cursor requests the postgres database."""
35-
janitor = DatabaseJanitor("user", "host", "1234", "database_name", 10, "some_password")
39+
janitor = DatabaseJanitor(
40+
user="user",
41+
host="host",
42+
port="1234",
43+
dbname="database_name",
44+
version=10,
45+
password="some_password",
46+
)
3647
with janitor.cursor():
3748
connect_mock.assert_called_once_with(
3849
dbname="postgres", user="user", password="some_password", host="host", port="1234"

0 commit comments

Comments
 (0)