-
Notifications
You must be signed in to change notification settings - Fork 192
/
Copy pathsetup.py
342 lines (265 loc) · 12.5 KB
/
setup.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
# -*- coding: utf-8 -*-
###########################################################################
# Copyright (c), The AiiDA team. All rights reserved. #
# This file is part of the AiiDA code. #
# #
# The code is hosted on GitHub at https://github.com/aiidateam/aiida-core #
# For further information on the license, see the LICENSE.txt file #
# For further information please visit http://www.aiida.net #
###########################################################################
"""Reusable command line interface options for the setup commands."""
import functools
import getpass
import hashlib
import click
from aiida.backends import BACKEND_DJANGO
from aiida.cmdline.params import options, types
from aiida.manage.configuration import get_config, get_config_option, Profile
from aiida.manage.external.postgres import DEFAULT_DBINFO
from aiida.manage.external.rmq import BROKER_DEFAULTS
PASSWORD_UNCHANGED = '***' # noqa
def validate_profile_parameter(ctx):
"""Validate that the context contains the option `profile` and it contains a `Profile` instance.
:param ctx: click context which should contain the selected profile
:raises: BadParameter if the context does not contain a `Profile` instance for option `profile`
"""
option = 'profile'
if option not in ctx.params or ctx.params[option] is None or not isinstance(ctx.params[option], Profile):
raise click.BadParameter('specifying the name of the profile is required', param_hint=f'"--{option}"')
def get_profile_attribute_default(attribute_tuple, ctx):
"""Return the default value for the given attribute of the profile passed in the context.
:param attribute: attribute for which to get the current value
:param ctx: click context which should contain the selected profile
:return: profile attribute default value if set, or None
"""
attribute, default = attribute_tuple
try:
validate_profile_parameter(ctx)
except click.BadParameter:
return default
else:
try:
return getattr(ctx.params['profile'], attribute)
except KeyError:
return default
def get_repository_uri_default(ctx):
"""Return the default value for the repository URI for the current profile in the click context.
:param ctx: click context which should contain the selected profile
:return: default repository URI
"""
import os
from aiida.manage.configuration.settings import AIIDA_CONFIG_FOLDER
validate_profile_parameter(ctx)
return os.path.join(AIIDA_CONFIG_FOLDER, 'repository', ctx.params['profile'].name)
def get_quicksetup_repository_uri(ctx, param, value): # pylint: disable=unused-argument
"""Return the repository URI to be used as default in `verdi quicksetup`
:param ctx: click context which should contain the contextual parameters
:return: the repository URI
"""
return get_repository_uri_default(ctx)
def get_quicksetup_database_name(ctx, param, value): # pylint: disable=unused-argument
"""Determine the database name to be used as default for the Postgres connection in `verdi quicksetup`
If a value is explicitly passed, that value is returned unchanged.
If no value is passed, the name will be <profile_name>_<os_user>_<hash>, where <os_user> is the name of the current
operating system user and <hash> is a hash of the path of the configuration directory.
Note: This ensures that profiles named ``test_...`` will have databases named ``test_...`` .
:param ctx: click context which should contain the contextual parameters
:return: the database name
"""
if value is not None:
return value
config = get_config()
profile = ctx.params['profile'].name
config_hash = hashlib.md5(config.dirpath.encode('utf-8')).hexdigest()
database_name = f'{profile}_{getpass.getuser()}_{config_hash}'
return database_name
def get_quicksetup_username(ctx, param, value): # pylint: disable=unused-argument
"""Determine the username to be used as default for the Postgres connection in `verdi quicksetup`
If a value is explicitly passed, that value is returned. If there is no value, the name will be based on the
name of the current operating system user and the hash of the path of the configuration directory.
:param ctx: click context which should contain the contextual parameters
:return: the username
"""
if value is not None:
return value
config = get_config()
config_hash = hashlib.md5(config.dirpath.encode('utf-8')).hexdigest()
username = f'aiida_qs_{getpass.getuser()}_{config_hash}'
return username
def get_quicksetup_password(ctx, param, value): # pylint: disable=unused-argument
"""Determine the password to be used as default for the Postgres connection in `verdi quicksetup`
If a value is explicitly passed, that value is returned. If there is no value, the current username in the context
will be scanned for in currently existing profiles. If it does, the corresponding password will be used. If no such
user already exists, a random password will be generated.
:param ctx: click context which should contain the contextual parameters
:return: the password
"""
from aiida.common.hashing import get_random_string
if value is not None:
return value
username = ctx.params['db_username']
config = get_config()
for available_profile in config.profiles:
if available_profile.database_username == username:
value = available_profile.database_password
break
else:
value = get_random_string(16)
return value
SETUP_PROFILE = options.OverridableOption(
'--profile',
prompt='Profile name',
help='The name of the new profile.',
required=True,
type=types.ProfileParamType(cannot_exist=True),
cls=options.interactive.InteractiveOption
)
SETUP_USER_EMAIL = options.USER_EMAIL.clone(
prompt='Email Address (for sharing data)',
default=get_config_option('autofill.user.email'),
required_fn=lambda x: get_config_option('autofill.user.email') is None,
required=True,
cls=options.interactive.InteractiveOption
)
SETUP_USER_FIRST_NAME = options.USER_FIRST_NAME.clone(
prompt='First name',
default=get_config_option('autofill.user.first_name'),
required_fn=lambda x: get_config_option('autofill.user.first_name') is None,
required=True,
cls=options.interactive.InteractiveOption
)
SETUP_USER_LAST_NAME = options.USER_LAST_NAME.clone(
prompt='Last name',
default=get_config_option('autofill.user.last_name'),
required_fn=lambda x: get_config_option('autofill.user.last_name') is None,
required=True,
cls=options.interactive.InteractiveOption
)
SETUP_USER_INSTITUTION = options.USER_INSTITUTION.clone(
prompt='Institution',
default=get_config_option('autofill.user.institution'),
required_fn=lambda x: get_config_option('autofill.user.institution') is None,
required=True,
cls=options.interactive.InteractiveOption
)
QUICKSETUP_DATABASE_ENGINE = options.DB_ENGINE
QUICKSETUP_DATABASE_BACKEND = options.DB_BACKEND
QUICKSETUP_DATABASE_HOSTNAME = options.DB_HOST
QUICKSETUP_DATABASE_PORT = options.DB_PORT
QUICKSETUP_DATABASE_NAME = options.OverridableOption(
'--db-name',
help='Name of the database to create.',
type=types.NonEmptyStringParamType(),
callback=get_quicksetup_database_name
)
QUICKSETUP_DATABASE_USERNAME = options.DB_USERNAME.clone(
help='Name of the database user to create.', callback=get_quicksetup_username
)
QUICKSETUP_DATABASE_PASSWORD = options.DB_PASSWORD.clone(callback=get_quicksetup_password)
QUICKSETUP_SUPERUSER_DATABASE_USERNAME = options.OverridableOption(
'--su-db-username', help='User name of the database super user.', type=click.STRING, default=DEFAULT_DBINFO['user']
)
QUICKSETUP_SUPERUSER_DATABASE_NAME = options.OverridableOption(
'--su-db-name',
help='Name of the template database to connect to as the database superuser.',
type=click.STRING,
default=DEFAULT_DBINFO['database']
)
QUICKSETUP_SUPERUSER_DATABASE_PASSWORD = options.OverridableOption(
'--su-db-password',
help='Password to connect as the database superuser.',
type=click.STRING,
hide_input=True,
default=DEFAULT_DBINFO['password'],
)
QUICKSETUP_BROKER_PROTOCOL = options.BROKER_PROTOCOL
QUICKSETUP_BROKER_USERNAME = options.BROKER_USERNAME
QUICKSETUP_BROKER_PASSWORD = options.BROKER_PASSWORD
QUICKSETUP_BROKER_HOST = options.BROKER_HOST
QUICKSETUP_BROKER_PORT = options.BROKER_PORT
QUICKSETUP_BROKER_VIRTUAL_HOST = options.BROKER_VIRTUAL_HOST
QUICKSETUP_REPOSITORY_URI = options.REPOSITORY_PATH.clone(
callback=get_quicksetup_repository_uri # Cannot use `default` because `ctx` is needed to determine the default
)
SETUP_DATABASE_ENGINE = QUICKSETUP_DATABASE_ENGINE.clone(
prompt='Database engine',
contextual_default=functools.partial(get_profile_attribute_default, ('database_engine', 'postgresql_psycopg2')),
cls=options.interactive.InteractiveOption
)
SETUP_DATABASE_BACKEND = QUICKSETUP_DATABASE_BACKEND.clone(
prompt='Database backend',
contextual_default=functools.partial(get_profile_attribute_default, ('database_backend', BACKEND_DJANGO)),
cls=options.interactive.InteractiveOption
)
SETUP_DATABASE_HOSTNAME = QUICKSETUP_DATABASE_HOSTNAME.clone(
prompt='Database host',
contextual_default=functools.partial(get_profile_attribute_default, ('database_hostname', DEFAULT_DBINFO['host'])),
cls=options.interactive.InteractiveOption
)
SETUP_DATABASE_PORT = QUICKSETUP_DATABASE_PORT.clone(
prompt='Database port',
contextual_default=functools.partial(get_profile_attribute_default, ('database_port', DEFAULT_DBINFO['port'])),
cls=options.interactive.InteractiveOption
)
SETUP_DATABASE_NAME = QUICKSETUP_DATABASE_NAME.clone(
prompt='Database name',
required=True,
contextual_default=functools.partial(get_profile_attribute_default, ('database_name', None)),
cls=options.interactive.InteractiveOption
)
SETUP_DATABASE_USERNAME = QUICKSETUP_DATABASE_USERNAME.clone(
prompt='Database username',
required=True,
contextual_default=functools.partial(get_profile_attribute_default, ('database_username', None)),
cls=options.interactive.InteractiveOption
)
SETUP_DATABASE_PASSWORD = QUICKSETUP_DATABASE_PASSWORD.clone(
prompt='Database password',
required=True,
contextual_default=functools.partial(get_profile_attribute_default, ('database_password', None)),
cls=options.interactive.InteractiveOption
)
SETUP_BROKER_PROTOCOL = QUICKSETUP_BROKER_PROTOCOL.clone(
prompt='Broker protocol',
required=True,
contextual_default=functools.partial(get_profile_attribute_default, ('broker_protocol', BROKER_DEFAULTS.protocol)),
cls=options.interactive.InteractiveOption
)
SETUP_BROKER_USERNAME = QUICKSETUP_BROKER_USERNAME.clone(
prompt='Broker username',
required=True,
contextual_default=functools.partial(get_profile_attribute_default, ('broker_username', BROKER_DEFAULTS.username)),
cls=options.interactive.InteractiveOption
)
SETUP_BROKER_PASSWORD = QUICKSETUP_BROKER_PASSWORD.clone(
prompt='Broker password',
required=True,
contextual_default=functools.partial(get_profile_attribute_default, ('broker_password', BROKER_DEFAULTS.password)),
cls=options.interactive.InteractiveOption
)
SETUP_BROKER_HOST = QUICKSETUP_BROKER_HOST.clone(
prompt='Broker host',
required=True,
contextual_default=functools.partial(get_profile_attribute_default, ('broker_host', BROKER_DEFAULTS.host)),
cls=options.interactive.InteractiveOption
)
SETUP_BROKER_PORT = QUICKSETUP_BROKER_PORT.clone(
prompt='Broker port',
required=True,
contextual_default=functools.partial(get_profile_attribute_default, ('broker_port', BROKER_DEFAULTS.port)),
cls=options.interactive.InteractiveOption
)
SETUP_BROKER_VIRTUAL_HOST = QUICKSETUP_BROKER_VIRTUAL_HOST.clone(
prompt='Broker virtual host name',
required=True,
contextual_default=functools.partial(
get_profile_attribute_default, ('broker_virtual_host', BROKER_DEFAULTS.virtual_host)
),
cls=options.interactive.InteractiveOption
)
SETUP_REPOSITORY_URI = QUICKSETUP_REPOSITORY_URI.clone(
prompt='Repository directory',
callback=None, # Unset the `callback` to define the default, which is instead done by the `contextual_default`
contextual_default=get_repository_uri_default,
cls=options.interactive.InteractiveOption
)