@@ -26,8 +26,9 @@ def __init__(
26
26
user : str ,
27
27
host : str ,
28
28
port : Union [str , int ],
29
- dbname : str ,
30
29
version : Union [str , float , Version ], # type: ignore[valid-type]
30
+ dbname : Optional [str ] = None ,
31
+ template_dbname : Optional [str ] = None ,
31
32
password : Optional [str ] = None ,
32
33
isolation_level : "Optional[psycopg.IsolationLevel]" = None ,
33
34
connection_timeout : int = 60 ,
@@ -38,6 +39,7 @@ def __init__(
38
39
:param host: postgresql host
39
40
:param port: postgresql port
40
41
:param dbname: database name
42
+ :param dbname: template database name
41
43
:param version: postgresql version number
42
44
:param password: optional postgresql password
43
45
:param isolation_level: optional postgresql isolation level
@@ -49,7 +51,10 @@ def __init__(
49
51
self .password = password
50
52
self .host = host
51
53
self .port = port
54
+ # At least one of the dbname or template_dbname has to be filled.
55
+ assert any ([dbname , template_dbname ])
52
56
self .dbname = dbname
57
+ self .template_dbname = template_dbname
53
58
self ._connection_timeout = connection_timeout
54
59
self .isolation_level = isolation_level
55
60
if not isinstance (version , Version ):
@@ -59,36 +64,33 @@ def __init__(
59
64
60
65
def init (self ) -> None :
61
66
"""Create database in postgresql."""
62
- template_name = f"{ self .dbname } _tmpl"
63
67
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 :
75
71
cur .execute (f'CREATE DATABASE "{ self .dbname } ";' )
76
72
else :
77
73
# All template database does not allow connection:
78
- self ._dont_datallowconn (cur , template_name )
74
+ self ._dont_datallowconn (cur , self . template_dbname )
79
75
# 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
83
83
84
84
def drop (self ) -> None :
85
85
"""Drop database in postgresql."""
86
86
# We cannot drop the database while there are connections to it, so we
87
87
# 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
88
90
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 } ";' )
92
94
93
95
@staticmethod
94
96
def _dont_datallowconn (cur : Cursor , dbname : str ) -> None :
@@ -113,12 +115,13 @@ def load(self, load: Union[Callable, str, Path]) -> None:
113
115
* a callable that expects: host, port, user, dbname and password arguments.
114
116
115
117
"""
118
+ db_to_load = self .template_dbname if self .is_template () else self .dbname
116
119
_loader = build_loader (load )
117
120
_loader (
118
121
host = self .host ,
119
122
port = self .port ,
120
123
user = self .user ,
121
- dbname = self . dbname ,
124
+ dbname = db_to_load ,
122
125
password = self .password ,
123
126
)
124
127
0 commit comments