###########################################################################
# 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 #
###########################################################################
"""``verdi presto`` command."""
from __future__ import annotations
import pathlib
import re
import typing as t
import click
from aiida.cmdline.commands.cmd_verdi import verdi
from aiida.cmdline.params import options
from aiida.cmdline.utils import echo
from aiida.manage.configuration import get_config_option
DEFAULT_PROFILE_NAME_PREFIX: str = 'presto'
[docs]
def get_default_presto_profile_name():
from aiida.common.exceptions import ConfigurationError
from aiida.manage import get_config
try:
profile_names = get_config().profile_names
except ConfigurationError:
# This can happen when tab-completing in an environment that did not create the configuration folder yet.
# It would have been possible to just call ``get_config(create=True)`` to create the config directory, but this
# should not be done during tab-completion just to generate a default value.
return DEFAULT_PROFILE_NAME_PREFIX
indices = []
for profile_name in profile_names:
if match := re.search(r'presto[-]?(\d+)?', profile_name):
indices.append(int(match.group(1) or '0'))
if not indices:
return DEFAULT_PROFILE_NAME_PREFIX
last_index = int(sorted(indices)[-1])
return f'{DEFAULT_PROFILE_NAME_PREFIX}-{last_index + 1}'
[docs]
def detect_postgres_config(
profile_name: str,
postgres_hostname: str,
postgres_port: int,
postgres_username: str,
postgres_password: str,
non_interactive: bool,
) -> dict[str, t.Any]:
"""Attempt to connect to the given PostgreSQL server and create a new user and database.
:raises ConnectionError: If no connection could be established to the PostgreSQL server or a user and database
could not be created.
:returns: The connection configuration for the newly created user and database as can be used directly for the
storage configuration of the ``core.psql_dos`` storage plugin.
"""
import secrets
from aiida.manage.configuration.settings import AIIDA_CONFIG_FOLDER
from aiida.manage.external.postgres import Postgres
dbinfo = {
'host': postgres_hostname,
'port': postgres_port,
'user': postgres_username,
'password': postgres_password,
}
postgres = Postgres(interactive=not non_interactive, quiet=False, dbinfo=dbinfo)
if not postgres.is_connected:
raise ConnectionError(f'Failed to connect to the PostgreSQL server using parameters: {dbinfo}')
database_name = f'aiida-{profile_name}'
database_username = f'aiida-{profile_name}'
database_password = secrets.token_hex(15)
try:
database_username, database_name = postgres.create_dbuser_db_safe(
dbname=database_name, dbuser=database_username, dbpass=database_password
)
except Exception as exception:
raise ConnectionError(f'Unable to automatically create the PostgreSQL user and database: {exception}')
return {
'database_hostname': postgres_hostname,
'database_port': postgres_port,
'database_name': database_name,
'database_username': database_username,
'database_password': database_password,
'repository_uri': f'file://{AIIDA_CONFIG_FOLDER / "repository" / profile_name}',
}
@verdi.command('presto')
@click.option(
'-p',
'--profile-name',
default=lambda: get_default_presto_profile_name(),
show_default=True,
help=f'Name of the profile. By default, a unique name starting with `{DEFAULT_PROFILE_NAME_PREFIX}` is '
'automatically generated.',
)
@click.option(
'--email',
default=lambda: get_config_option('autofill.user.email') or 'aiida@localhost',
show_default=True,
help='Email of the default user.',
)
@click.option(
'--use-postgres',
is_flag=True,
help='When toggled on, the profile uses a PostgreSQL database instead of an SQLite one. The connection details to '
'the PostgreSQL server can be configured with the relevant options. The command attempts to automatically create a '
'user and database to use for the profile, but this can fail depending on the configuration of the server.',
)
@click.option('--postgres-hostname', type=str, default='localhost', help='The hostname of the PostgreSQL server.')
@click.option('--postgres-port', type=int, default=5432, help='The port of the PostgreSQL server.')
@click.option(
'--postgres-username',
type=str,
default='postgres',
help='The username of the PostgreSQL user that is authorized to create new databases.',
)
@click.option(
'--postgres-password',
type=str,
required=False,
help='The password of the PostgreSQL user that is authorized to create new databases.',
)
@options.NON_INTERACTIVE(help='Never prompt, such as for sudo password.')
@click.pass_context
def verdi_presto(
ctx,
profile_name,
email,
use_postgres,
postgres_hostname,
postgres_port,
postgres_username,
postgres_password,
non_interactive,
):
"""Set up a new profile in a jiffy.
This command aims to make setting up a new profile as easy as possible. It does not require any services, such as
PostgreSQL and RabbitMQ. It intentionally provides only a limited amount of options to customize the profile and by
default does not require any options to be specified at all. To create a new profile with full control over its
configuration, please use `verdi profile setup` instead.
After running `verdi presto` you can immediately start using AiiDA without additional setup. The command performs
the following actions:
\b
* Create a new profile that is set as the new default
* Create a default user for the profile (email can be configured through the `--email` option)
* Set up the localhost as a `Computer` and configure it
* Set a number of configuration options with sensible defaults
By default the command creates a profile that uses SQLite for the database. It automatically checks for RabbitMQ
running on the localhost, and, if it can connect, configures that as the broker for the profile. Otherwise, the
profile is created without a broker, in which case some functionality will be unavailable, most notably running the
daemon and submitting processes to said daemon.
When the `--use-postgres` flag is toggled, the command tries to connect to the PostgreSQL server with connection
paramaters taken from the `--postgres-hostname`, `--postgres-port`, `--postgres-username` and `--postgres-password`
options. It uses these credentials to try and automatically create a user and database. If successful, the newly
created profile uses the new PostgreSQL database instead of SQLite.
"""
from aiida.brokers.rabbitmq.defaults import detect_rabbitmq_config
from aiida.common import docs, exceptions
from aiida.manage.configuration import create_profile, load_profile
from aiida.orm import Computer
if profile_name in ctx.obj.config.profile_names:
raise click.BadParameter(f'The profile `{profile_name}` already exists.', param_hint='--profile-name')
postgres_config_kwargs = {
'profile_name': profile_name,
'postgres_hostname': postgres_hostname,
'postgres_port': postgres_port,
'postgres_username': postgres_username,
'postgres_password': postgres_password,
'non_interactive': non_interactive,
}
storage_backend: str = 'core.sqlite_dos'
storage_config: dict[str, t.Any] = {}
if use_postgres:
try:
storage_config = detect_postgres_config(**postgres_config_kwargs)
except ConnectionError as exception:
echo.echo_critical(str(exception))
else:
echo.echo_report(
'`--use-postgres` enabled and database creation successful: configuring the profile to use PostgreSQL.'
)
storage_backend = 'core.psql_dos'
else:
echo.echo_report('Option `--use-postgres` not enabled: configuring the profile to use SQLite.')
broker_backend = None
broker_config = None
try:
broker_config = detect_rabbitmq_config()
except ConnectionError as exception:
echo.echo_report(f'RabbitMQ server not found ({exception}): configuring the profile without a broker.')
echo.echo_report(f'See {docs.URL_NO_BROKER} for details on the limitations of running without a broker.')
else:
echo.echo_report('RabbitMQ server detected: configuring the profile with a broker.')
broker_backend = 'core.rabbitmq'
try:
profile = create_profile(
ctx.obj.config,
name=profile_name,
email=email,
storage_backend=storage_backend,
storage_config=storage_config,
broker_backend=broker_backend,
broker_config=broker_config,
)
except (ValueError, TypeError, exceptions.EntryPointError, exceptions.StorageMigrationError) as exception:
echo.echo_critical(str(exception))
echo.echo_success(f'Created new profile `{profile.name}`.')
ctx.obj.config.set_option('runner.poll.interval', 1, scope=profile.name)
ctx.obj.config.set_default_profile(profile.name, overwrite=True)
ctx.obj.config.store()
load_profile(profile.name, allow_switch=True)
echo.echo_info(f'Loaded newly created profile `{profile.name}`.')
filepath_scratch = pathlib.Path(ctx.obj.config.dirpath) / 'scratch' / profile.name
computer = Computer(
label='localhost',
hostname='localhost',
description='Localhost automatically created by `verdi presto`',
transport_type='core.local',
scheduler_type='core.direct',
workdir=str(filepath_scratch),
).store()
computer.configure(safe_interval=0)
computer.set_minimum_job_poll_interval(1)
computer.set_default_mpiprocs_per_machine(1)
echo.echo_success('Configured the localhost as a computer.')