Source code for aiida.cmdline.commands.cmd_daemon

###########################################################################
# 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 daemon` commands."""

from __future__ import annotations

import subprocess
import sys
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 decorators, echo


[docs] def validate_daemon_workers(ctx, param, value): """Validate the value for the number of daemon workers to start with default set by config.""" if value is None: value = ctx.obj.config.get_option('daemon.default_workers', ctx.obj.profile.name) if not isinstance(value, int): raise click.BadParameter(f'{value} is not an integer') if value <= 0: raise click.BadParameter(f'{value} is not a positive non-zero integer') return value
@verdi.group('daemon') def verdi_daemon(): """Inspect and manage the daemon."""
[docs] def execute_client_command(command: str, daemon_not_running_ok: bool = False, **kwargs) -> dict[str, t.Any] | None: """Execute a command of the :class:`aiida.engine.daemon.client.DaemonClient` and echo whether it failed or not. :param command: The name of hte method. :param daemon_not_running_ok: If ``True``, the command raising an exception because the daemon was not running is not treated as a failure. :param kwargs: Keyword arguments that are passed to the client method. """ from aiida.common.exceptions import ConfigurationError from aiida.engine.daemon.client import DaemonException, DaemonNotRunningException, get_daemon_client try: client = get_daemon_client() except ConfigurationError: echo.echo('WARNING', fg=echo.COLORS['warning'], bold=True) return None try: response = getattr(client, command)(**kwargs) except DaemonNotRunningException as exception: if daemon_not_running_ok: echo.echo('OK', fg=echo.COLORS['success'], bold=True) else: echo.echo('FAILED', fg=echo.COLORS['error'], bold=True) echo.echo_critical(str(exception)) except DaemonException as exception: echo.echo('FAILED', fg=echo.COLORS['error'], bold=True) echo.echo_critical(str(exception)) else: echo.echo('OK', fg=echo.COLORS['success'], bold=True) return response return None
@verdi_daemon.command() @click.option('--foreground', is_flag=True, help='Run in foreground.') @click.argument('number', required=False, type=int, callback=validate_daemon_workers) @options.TIMEOUT(default=None, required=False, type=int) @decorators.with_dbenv() @decorators.requires_broker @decorators.check_circus_zmq_version def start(foreground, number, timeout): """Start the daemon with NUMBER workers. If the NUMBER of desired workers is not specified, the default is used, which is determined by the configuration option `daemon.default_workers`, which if not explicitly changed defaults to 1. Returns exit code 0 if the daemon is OK, non-zero if there was an error. """ echo.echo(f'Starting the daemon with {number} workers... ', nl=False) execute_client_command('start_daemon', number_workers=number, foreground=foreground, timeout=timeout) @verdi_daemon.command() @click.option('--all', 'all_profiles', is_flag=True, help='Show status of all daemons.') @options.TIMEOUT(default=None, required=False, type=int) @click.pass_context @decorators.requires_loaded_profile() @decorators.requires_broker def status(ctx, all_profiles, timeout): """Print the status of the current daemon or all daemons. Returns exit code 0 if all requested daemons are running, else exit code 3. """ from tabulate import tabulate from aiida.cmdline.utils.common import format_local_time from aiida.engine.daemon.client import DaemonException, get_daemon_client if all_profiles is True: profiles = [profile for profile in ctx.obj.config.profiles if not profile.is_test_profile] else: profiles = [ctx.obj.profile] daemons_running = [] for profile in profiles: client = get_daemon_client(profile.name) echo.echo('Profile: ', fg=echo.COLORS['report'], bold=True, nl=False) echo.echo(f'{profile.name}', bold=True) try: client.get_status(timeout=timeout) except DaemonException as exception: echo.echo_error(str(exception)) daemons_running.append(False) continue worker_response = client.get_worker_info() daemon_response = client.get_daemon_info() workers = [] for pid, info in worker_response['info'].items(): if isinstance(info, dict): row = [pid, info['mem'], info['cpu'], format_local_time(info['create_time'])] else: row = [pid, '-', '-', '-'] workers.append(row) if workers: workers_info = tabulate(workers, headers=['PID', 'MEM %', 'CPU %', 'started'], tablefmt='simple') else: workers_info = '--> No workers are running. Use `verdi daemon incr` to start some!\n' start_time = format_local_time(daemon_response['info']['create_time']) echo.echo( f'Daemon is running as PID {daemon_response["info"]["pid"]} since {start_time}\n' f'Active workers [{len(workers)}]:\n{workers_info}\n' 'Use `verdi daemon [incr | decr] [num]` to increase / decrease the number of workers' ) if not all(daemons_running): sys.exit(3) @verdi_daemon.command() @click.argument('number', default=1, type=int) @options.TIMEOUT(default=None, required=False, type=int) @decorators.requires_broker @decorators.only_if_daemon_running() def incr(number, timeout): """Add NUMBER [default=1] workers to the running daemon. Returns exit code 0 if the daemon is OK, non-zero if there was an error. """ echo.echo(f'Starting {number} daemon workers... ', nl=False) execute_client_command('increase_workers', number=number, timeout=timeout) @verdi_daemon.command() @click.argument('number', default=1, type=int) @options.TIMEOUT(default=None, required=False, type=int) @decorators.requires_broker @decorators.only_if_daemon_running() def decr(number, timeout): """Remove NUMBER [default=1] workers from the running daemon. Returns exit code 0 if the daemon is OK, non-zero if there was an error. """ echo.echo(f'Stopping {number} daemon workers... ', nl=False) execute_client_command('decrease_workers', number=number, timeout=timeout) @verdi_daemon.command() def logshow(): """Show the log of the daemon, press CTRL+C to quit.""" from aiida.engine.daemon.client import get_daemon_client client = get_daemon_client() with subprocess.Popen(['tail', '-f', client.daemon_log_file], env=client.get_env()) as process: process.wait() @verdi_daemon.command() @click.option('--no-wait', is_flag=True, help='Do not wait for confirmation.') @click.option('--all', 'all_profiles', is_flag=True, help='Stop all daemons.') @options.TIMEOUT(default=None, required=False, type=int) @decorators.requires_broker @click.pass_context def stop(ctx, no_wait, all_profiles, timeout): """Stop the daemon. Returns exit code 0 if the daemon was shut down successfully (or was not running), non-zero if there was an error. """ if all_profiles is True: profiles = [profile for profile in ctx.obj.config.profiles if not profile.is_test_profile] else: profiles = [ctx.obj.profile] for profile in profiles: echo.echo('Profile: ', fg=echo.COLORS['report'], bold=True, nl=False) echo.echo(f'{profile.name}', bold=True) echo.echo('Stopping the daemon... ', nl=False) execute_client_command('stop_daemon', daemon_not_running_ok=True, wait=not no_wait, timeout=timeout) @verdi_daemon.command() @click.option('--reset', is_flag=True, help='Completely reset the daemon.') @click.option('--no-wait', is_flag=True, help='Do not wait for confirmation.') @options.TIMEOUT(default=None, required=False, type=int) @click.pass_context @decorators.with_dbenv() @decorators.requires_broker @decorators.only_if_daemon_running() def restart(ctx, reset, no_wait, timeout): """Restart the daemon. The daemon is stopped before being restarted with the default number of workers that is started when calling `verdi daemon start` manually. Returns exit code 0 if the result is OK, non-zero if there was an error. """ if reset: echo.echo_deprecated( '`--reset` flag is deprecated. Now, `verdi daemon restart` by default restarts the full daemon.' ) if no_wait: echo.echo_deprecated('The `--no-wait` flag is deprecated and no longer has any effect.') if timeout is not None: echo.echo_deprecated('The `--timeout` option is deprecated and no longer has any effect.') # These two lines can be simplified to `ctx.invoke(start)` once issue #950 in `click` is resolved. # Due to that bug, the `callback` of the `number` argument the `start` command is not being called, which is # responsible for settting the default value, which causes `None` to be passed and that triggers an exception. # As a temporary workaround, we fetch the default here manually and pass that in explicitly. number = ctx.obj.config.get_option('daemon.default_workers', ctx.obj.profile.name) ctx.invoke(stop) ctx.invoke(start, number=number) @verdi_daemon.command(hidden=True) @click.option('--foreground', is_flag=True, help='Run in foreground.') @click.argument('number', required=False, type=int, callback=validate_daemon_workers) @decorators.with_dbenv() @decorators.requires_broker @decorators.check_circus_zmq_version def start_circus(foreground, number): """This will actually launch the circus daemon, either daemonized in the background or in the foreground. If run in the foreground all logs are redirected to stdout. .. note:: this should not be called directly from the commandline! """ from aiida.engine.daemon.client import get_daemon_client get_daemon_client()._start_daemon(number_workers=number, foreground=foreground) @verdi_daemon.command('worker') @decorators.with_dbenv() @decorators.requires_broker def worker(): """Run a single daemon worker in the current interpreter.""" from aiida.engine.daemon.worker import start_daemon_worker start_daemon_worker(foreground=True)