Source code for aiida.cmdline.commands.cmd_presto

###########################################################################
# 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.')