Source code for aiida.cmdline.verdilib

# -*- 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               #
###########################################################################
"""
Command line commands for the main executable 'verdi' of aiida

If you want to define a new command line parameter, just define a new
class inheriting from VerdiCommand, and define a run(self,*args) method
accepting a variable-length number of parameters args
(the command-line parameters), which will be invoked when
this executable is called as
verdi NAME

Don't forget to add the docstring to the class: the first line will be the
short description, the following ones the long description.
"""
import sys
import os
import contextlib
import click

import aiida
import aiida.cmdline.commands.user
import aiida.control.user
from aiida.common.exceptions import (
    AiidaException, ConfigurationError, ProfileConfigurationError)
from aiida.cmdline.baseclass import VerdiCommand, VerdiCommandRouter
from aiida.cmdline import pass_to_django_manage
from aiida.backends import settings as settings_profile
from aiida.control.postgres import Postgres, manual_setup_instructions, prompt_db_info
from aiida.cmdline.commands import verdi
from  aiida.backends.profile import (BACKEND_DJANGO, BACKEND_SQLA)

# Import here from other files; once imported, it will be found and
# used as a command-line parameter
from aiida.cmdline.commands.user import User
import aiida.cmdline.commands.user as user
from aiida.cmdline.commands.calculation import Calculation
from aiida.cmdline.commands.code import Code
from aiida.cmdline.commands.computer import Computer
from aiida.cmdline.commands.daemon import Daemon
from aiida.cmdline.commands.data import Data
from aiida.cmdline.commands.devel import Devel
from aiida.cmdline.commands.exportfile import Export
from aiida.cmdline.commands.group import Group
from aiida.cmdline.commands.graph import Graph
from aiida.cmdline.commands.importfile import Import
from aiida.cmdline.commands.node import Node
from aiida.cmdline.commands.profile import Profile
from aiida.cmdline.commands.workflow import Workflow
from aiida.cmdline.commands.work import Work
from aiida.cmdline.commands.comment import Comment
from aiida.cmdline.commands.shell import Shell
from aiida.cmdline.commands.restapi import Restapi
from aiida.cmdline import execname


[docs]class ProfileParsingException(AiidaException): """ Exception raised when parsing the profile command line option, if only -p is provided, and no profile is specified """ def __init__(self, *args, **kwargs): self.minus_p_provided = kwargs.pop('minus_p_provided', False) super(ProfileParsingException, self).__init__(*args, **kwargs)
[docs]def parse_profile(argv, merge_equal=False): """ Parse the argv to see if a profile has been specified, return it with the command position shift (index where the commands start) :param merge_equal: if True, merge things like ('verdi', '--profile', '=', 'x', 'y') to ('verdi', '--profile=x', 'y') but then return the correct index for the original array. :raise ProfileParsingException: if there is only 'verdi' specified, or if only 'verdi -p' (in these cases, one has respectively exception.minus_p_provided equal to False or True) """ if merge_equal: if len(argv) >= 3: if argv[1] == '--profile' and argv[2] == '=': internal_argv = [argv[0], "".join(argv[1:4])] + list(argv[4:]) shift = 2 else: internal_argv = list(argv) shift = 0 else: internal_argv = list(argv) shift = 0 else: internal_argv = list(argv) shift = 0 profile = None # Use default profile if nothing is specified command_position = 1 # If there is no profile option try: profile_switch = internal_argv[1] except IndexError: raise ProfileParsingException(minus_p_provided=False) long_profile_prefix = '--profile=' if profile_switch == '-p': try: profile = internal_argv[2] except IndexError: raise ProfileParsingException(minus_p_provided=True) command_position = 3 elif profile_switch.startswith(long_profile_prefix): profile = profile_switch[len(long_profile_prefix):] command_position = 2 else: # No profile switch, continue using argv[1] as the command name pass return profile, command_position + shift
@contextlib.contextmanager
[docs]def update_environment(new_argv): """ Used as a context manager, changes sys.argv with the new_argv argument, and restores it upon exit. """ import sys _argv = sys.argv[:] sys.argv = new_argv[:] yield # Restore old parameters when exiting from the context manager sys.argv = _argv
######################################################################## # HERE STARTS THE COMMAND FUNCTION LIST ########################################################################
[docs]class CompletionCommand(VerdiCommand): """ Return the bash completion function to put in ~/.bashrc This command prints on screen the function to be inserted in your .bashrc command. You can copy and paste the output, or simply add eval "`verdi completioncommand`" to your .bashrc, *AFTER* having added the aiida/bin directory to the path. """
[docs] def run(self, *args): """ I put the documentation here, and I don't print it, so we don't clutter too much the .bashrc. * "${THE_WORDS[@]}" (with the @) puts each element as a different parameter; note that the variable expansion etc. is performed * I add a 'x' at the end and then remove it; in this way, $( ) will not remove trailing spaces * If the completion command did not print anything, we use the default bash completion for filenames * If instead the code prints something empty, thanks to the workaround above $OUTPUT is not empty, so we do go the the 'else' case and then, no substitution is suggested. """ print r""" function _aiida_verdi_completion { OUTPUT=$( $1 completion "$COMP_CWORD" "${COMP_WORDS[@]}" ; echo 'x') OUTPUT=${OUTPUT%x} if [ -z "$OUTPUT" ] then # Only newline is a valid separator local IFS=$'\n' COMPREPLY=( $(compgen -o default -- "${COMP_WORDS[COMP_CWORD]}" ) ) # Add either a slash or a space, depending on whether it is a folder # or a file. printf %q escapes the filename if there are spaces. for ((i=0; i < ${#COMPREPLY[@]}; i++)); do [ -d "${COMPREPLY[$i]}" ] && \ COMPREPLY[$i]=$(printf %q%s "${COMPREPLY[$i]}" "/") || \ COMPREPLY[$i]=$(printf %q%s "${COMPREPLY[$i]}" " ") done else COMPREPLY=( $(compgen -W "$OUTPUT" -- "${COMP_WORDS[COMP_CWORD]}" ) ) # Always add a space after each command for ((i=0; i < ${#COMPREPLY[@]}; i++)); do COMPREPLY[$i]="${COMPREPLY[$i]} " done fi } complete -o nospace -F _aiida_verdi_completion verdi """
def complete(self, subargs_idx, subargs): # disable further completion print ""
[docs]class Completion(VerdiCommand): """ Manage bash completion Return a list of available commands, separated by spaces. Calls the correct function of the command if the TAB has been pressed after the first command. Returning without printing will use the default bash completion. """ # TODO: manage completion at a deeper level def run(self, *args): try: cword = int(args[0]) if cword <= 0: cword = 1 except IndexError: cword = 1 except ValueError: return try: profile, command_position = parse_profile(args[1:], merge_equal=True) except ProfileParsingException as e: cword_offset = 0 else: cword_offset = command_position - 1 if cword == 1 + cword_offset: print " ".join(sorted(short_doc.keys())) return else: try: # args[0] is cword; # args[1] is the executable (verdi) # args[2] is the command for verdi # args[3:] are the following subargs command = args[2 + cword_offset] except IndexError: return try: CommandClass = list_commands[command] except KeyError: return CommandClass().complete(subargs_idx=cword - 2 - cword_offset, subargs=args[3 + cword_offset:])
[docs]class ListParams(VerdiCommand): """ List available commands List available commands and their short description. For the long description, use the 'help' command. """ def run(self, *args): print get_listparams()
[docs]class Help(VerdiCommand): """ Describe a specific command Pass a further argument to get a description of a given command. """ def run(self, *args): try: command = args[0] except IndexError: print get_listparams() print "" print ( "Before each command you can specify the AiiDA profile to use," " with 'verdi -p <profile> <command>' or " "'verdi --profile=<profile> <command>'") print "" print ("Use '{} help <command>' for more information " "on a specific command.".format(execname)) sys.exit(1) if command in short_doc: print "Description for '%s %s'" % (execname, command) print "" print "**", short_doc[command] if command in long_doc: print long_doc[command] else: print >> sys.stderr, ( "{}: '{}' is not a valid command. " "See '{} help' for more help.".format( execname, command, execname)) get_command_suggestion(command) sys.exit(1) def complete(self, subargs_idx, subargs): if subargs_idx == 0: print " ".join(sorted(short_doc.keys())) else: print ""
[docs]class Install(VerdiCommand): """ Install/setup aiida for the current user This command creates the ~/.aiida folder in the home directory of the user, interactively asks for the database settings and the repository location, does a setup of the daemon and runs a migrate command to create/setup the database. """ def run(self, *args): click.echo('\nwarning: verdi install is deprecated, use verdi setup.\n') ctx = _setup_cmd.make_context('setup', list(args)) with ctx: _setup_cmd.invoke(ctx)
[docs] def complete(self, subargs_idx, subargs): """ No completion after 'verdi install'. """ print ""
[docs]class Setup(VerdiCommand): """ Setup aiida for the current user This command creates the ~/.aiida folder in the home directory of the user, interactively asks for the database settings and the repository location, does a setup of the daemon and runs a migrate command to create/setup the database. """ def run(self, *args): ctx = self._ctx(args) with ctx: _setup_cmd.invoke(ctx) @staticmethod def _ctx(args, info_name='verdi setup', **kwargs): return _setup_cmd.make_context(info_name, list(args), **kwargs)
[docs] def complete(self, subargs_idx, subargs): """ No completion after 'verdi install'. """ print ""
CONTEXT_SETTINGS = dict(help_option_names=['-h', '--help']) @verdi.command('setup', context_settings=CONTEXT_SETTINGS) @click.argument('profile', default='', type=str) @click.option('--only-config', is_flag=True) @click.option('--non-interactive', is_flag=True, help='never prompt the user for input, read values from options') @click.option('--backend', type=click.Choice(['django', 'sqlalchemy']),) @click.option('--email', type=str) @click.option('--db_host', type=str) @click.option('--db_port', type=int) @click.option('--db_name', type=str) @click.option('--db_user', type=str) @click.option('--db_pass', type=str) @click.option('--first-name', type=str) @click.option('--last-name', type=str) @click.option('--institution', type=str) @click.option('--no-password', is_flag=True) @click.option('--repo', type=str) def _setup_cmd(profile, only_config, non_interactive, backend, email, db_host, db_port, db_name, db_user, db_pass, first_name, last_name, institution, no_password, repo): '''verdi setup command, forward cmdline arguments to the setup function. Note: command line options are IGNORED unless --non-interactive is given.''' kwargs = dict( profile=profile, only_config=only_config, non_interactive=non_interactive, backend=backend, email=email, db_host=db_host, db_port=db_port, db_name=db_name, db_user=db_user, db_pass=db_pass, first_name=first_name, last_name=last_name, institution=institution, repo=repo ) kwargs = {k: v for k, v in kwargs.items() if v is not None} setup(**kwargs)
[docs]def setup(profile, only_config, non_interactive=False, **kwargs): ''' setup an aiida profile and aiida user (and the aiida default user). :param profile: Profile name :param only_config: do not create a new user :param non_interactive: do not prompt for configuration values, fail if not all values are given as kwargs. :param backend: one of 'django', 'sqlalchemy' :param email: valid email address for the user :param db_host: hostname for the database :param db_port: port to connect to the database :param db_user: name of the db user :param db_pass: password of the db user ''' from aiida.common.setup import (create_base_dirs, create_configuration, set_default_profile, DEFAULT_UMASK, create_config_noninteractive) from aiida.backends.profile import BACKEND_SQLA, BACKEND_DJANGO from aiida.backends.utils import set_backend_type from aiida.common.exceptions import InvalidOperation # ~ cmdline_args = list(args) # ~ only_user_config = False # ~ try: # ~ cmdline_args.remove('--only-config') # ~ only_user_config = True # ~ except ValueError: # ~ # Parameter not provided # ~ pass only_user_config = only_config # ~ if cmdline_args: # ~ print >> sys.stderr, "Unknown parameters on the command line: " # ~ print >> sys.stderr, ", ".join(cmdline_args) # ~ sys.exit(1) # create the directories to store the configuration files create_base_dirs() # gprofile = 'default' if profile is None else profile # ~ gprofile = profile if settings_profile.AIIDADB_PROFILE is None \ # ~ else settings_profile.AIIDADB_PROFILE if settings_profile.AIIDADB_PROFILE and profile: sys.exit('the profile argument cannot be used if verdi is called with -p option: {} and {}'.format( settings_profile.AIIDADB_PROFILE, profile)) gprofile = settings_profile.AIIDADB_PROFILE or profile if gprofile == profile: settings_profile.AIIDADB_PROFILE = profile if not settings_profile.AIIDADB_PROFILE: settings_profile.AIIDADB_PROFILE = 'default' # used internally later gprofile = settings_profile.AIIDADB_PROFILE created_conf = None # ask and store the configuration of the DB if non_interactive: try: created_conf = create_config_noninteractive( profile=gprofile, backend=kwargs['backend'], email=kwargs['email'], db_host=kwargs['db_host'], db_port=kwargs['db_port'], db_name=kwargs['db_name'], db_user=kwargs['db_user'], db_pass=kwargs.get('db_pass', ''), repo=kwargs['repo'], force_overwrite=kwargs.get('force_overwrite', False) ) except ValueError as e: click.echo("Error during configuation: {}".format(e.message), err=True) sys.exit(1) except KeyError as e: click.echo("--non-interactive requires all values to be given on the commandline! Missing argument: {}".format(e.message), err=True) sys.exit(1) else: try: created_conf = create_configuration(profile=gprofile) except ValueError as e: print >> sys.stderr, "Error during configuration: {}".format( e.message) sys.exit(1) # set default DB profiles set_default_profile('verdi', gprofile, force_rewrite=False) set_default_profile('daemon', gprofile, force_rewrite=False) if only_user_config: print ("Only user configuration requested, " "skipping the migrate command") else: print "Executing now a migrate command..." backend_choice = created_conf['AIIDADB_BACKEND'] if backend_choice == BACKEND_DJANGO: print("...for Django backend") # The correct profile is selected within load_dbenv. # Setting os.umask here since sqlite database gets created in # this step. old_umask = os.umask(DEFAULT_UMASK) # This check should be done more properly # try: # backend_type = get_backend_type() # except KeyError: # backend_type = None # # if backend_type is not None and backend_type != BACKEND_DJANGO: # raise InvalidOperation("An already existing database found" # "and a different than the selected" # "backend was used for its " # "management.") try: pass_to_django_manage([execname, 'migrate'], profile=gprofile) finally: os.umask(old_umask) set_backend_type(BACKEND_DJANGO) elif backend_choice == BACKEND_SQLA: print("...for SQLAlchemy backend") from aiida import is_dbenv_loaded from aiida.backends import settings from aiida.backends.sqlalchemy.utils import ( _load_dbenv_noschemacheck, check_schema_version) from aiida.backends.profile import load_profile # We avoid calling load_dbenv since we want to force the schema # migration if not is_dbenv_loaded(): settings.LOAD_DBENV_CALLED = True # This is going to set global variables in settings, including # settings.BACKEND load_profile() _load_dbenv_noschemacheck() # Perform the needed migration quietly check_schema_version(force_migration=True) set_backend_type(BACKEND_SQLA) else: raise InvalidOperation("Not supported backend selected.") print "Database was created successfully" # I create here the default user print "Loading new environment..." if only_user_config: from aiida.backends.utils import load_dbenv, is_dbenv_loaded # db environment has not been loaded in this case if not is_dbenv_loaded(): load_dbenv() from aiida.common.setup import DEFAULT_AIIDA_USER from aiida.orm.user import User as AiiDAUser if not AiiDAUser.search_for_users(email=DEFAULT_AIIDA_USER): print "Installing default AiiDA user..." nuser = AiiDAUser(email=DEFAULT_AIIDA_USER) nuser.first_name = "AiiDA" nuser.last_name = "Daemon" nuser.is_staff = True nuser.is_active = True nuser.is_superuser = True nuser.force_save() from aiida.common.utils import get_configured_user_email email = get_configured_user_email() print "Starting user configuration for {}...".format(email) if email == DEFAULT_AIIDA_USER: print "You set up AiiDA using the default Daemon email ({}),".format( email) print "therefore no further user configuration will be asked." else: # Ask to configure the new user if not non_interactive: user.configure.main(args=[email]) else: # or don't ask aiida.cmdline.commands.user.do_configure(kwargs['email'], kwargs.get('first_name'), kwargs.get('last_name'), kwargs.get('institution'), True) print "Setup finished."
[docs]class Quicksetup(VerdiCommand): ''' Quick setup for the most common usecase (1 user, 1 machine). Creates a database user 'aiida_qs_<login-name>' with random password (if it doesn't exist). Creates a database '<profile>_<username>' (if it exists, prompts user to use or change the name). ''' from aiida.backends.profile import (BACKEND_DJANGO, BACKEND_SQLA) def run(self, *args): ctx = self._ctx(args) with ctx: ctx.obj = self quicksetup.invoke(ctx) @staticmethod def _ctx(args, info_name='verdi quicksetup', **kwargs): return quicksetup.make_context(info_name, list(args), **kwargs)
@verdi.command('quicksetup', context_settings=CONTEXT_SETTINGS) @click.option('--profile', prompt='Profile name', type=str, default='quicksetup') @click.option('--email', prompt='Email Address (identifies your data when sharing)', type=str, help='This email address will be associated with your data and will be exported along with it, should you choose to share any of your work') @click.option('--first-name', prompt='First Name', type=str) @click.option('--last-name', prompt='Last Name', type=str) @click.option('--institution', prompt='Institution', type=str) @click.option('--backend', type=click.Choice([BACKEND_DJANGO, BACKEND_SQLA]), default=BACKEND_DJANGO) @click.option('--db-port', type=int) @click.option('--db-user', type=str) @click.option('--db-user-pw', type=str) @click.option('--db-name', type=str) @click.option('--repo', type=str) @click.option('--set-default/--no-set-default', default=None, help='Whether to set new profile as default for shell and daemon.') @click.option('--non-interactive', is_flag=True, help='never prompt the user for input, read values from options') @click.pass_obj def quicksetup(self, profile, email, first_name, last_name, institution, backend, db_port, db_user, db_user_pw, db_name, repo, set_default, non_interactive): '''Set up a sane aiida configuration with as little interaction as possible.''' from aiida.common.setup import create_base_dirs, AIIDA_CONFIG_FOLDER create_base_dirs() aiida_dir = os.path.expanduser(AIIDA_CONFIG_FOLDER) # access postgres postgres = Postgres(port=db_port, interactive=bool(not non_interactive), quiet=False) postgres.set_setup_fail_callback(prompt_db_info) success = postgres.determine_setup() if not success: sys.exit(1) # default database name is <profile>_<login-name> # this ensures that for profiles named test_... the database will also # be named test_... import getpass osuser = getpass.getuser() dbname = db_name or profile + '_' + osuser # default database user name is aiida_qs_<login-name> # default password is random dbuser = db_user or 'aiida_qs_' + osuser from aiida.common.setup import generate_random_secret_key dbpass = db_user_pw or generate_random_secret_key() # check if there is a profile that contains the db user already # and if yes, take the db user password from there # This is ok because a user can only see his own config files from aiida.common.setup import (set_default_profile, get_or_create_config) confs = get_or_create_config() profs = confs.get('profiles', {}) for v in profs.itervalues(): if v.get('AIIDADB_USER', '') == dbuser and not db_user_pw: dbpass = v.get('AIIDADB_PASS') print 'using found password for {}'.format(dbuser) break try: create = True if not postgres.dbuser_exists(dbuser): postgres.create_dbuser(dbuser, dbpass) else: dbname, create = _check_db_name(dbname, postgres) if create: postgres.create_db(dbuser, dbname) except Exception as e: click.echo('\n'.join([ 'Oops! Something went wrong while creating the database for you.', 'You may continue with the quicksetup, however:', 'For aiida to work correctly you will have to do that yourself as follows.', manual_setup_instructions(dbuser=dbuser, dbname=dbname), '', 'Or setup your (OS-level) user to have permissions to create databases and rerun quicksetup.', '' ])) raise e # create a profile, by default 'quicksetup' and prompt the user if # already exists confs = get_or_create_config() profile_name = profile or 'quicksetup' write_profile = False while not write_profile: if profile_name in confs.get('profiles', {}): if click.confirm('overwrite existing profile {}?'.format(profile_name)): write_profile = True else: profile_name = click.prompt('new profile name', type=str) else: write_profile = True dbhost = postgres.dbinfo.get('host', 'localhost') dbport = postgres.dbinfo.get('port', '5432') from os.path import isabs repo = repo or 'repository-{}/'.format(profile_name) if not isabs(repo): repo = os.path.join(aiida_dir, repo) setup_args = { 'backend': backend, 'email': email, 'db_host': dbhost, 'db_port': dbport, 'db_name': dbname, 'db_user': dbuser, 'db_pass': dbpass, 'repo': repo, 'first_name': first_name, 'last_name': last_name, 'institution': institution, 'force_overwrite': write_profile, } setup(profile_name, only_config=False, non_interactive=True, **setup_args) # Loop over all valid processes and check if a default profile is set for them # If not set the newly created profile as default, otherwise prompt whether to override from aiida.cmdline.commands.profile import valid_processes default_profiles = confs.get('default_profiles', {}) for process in valid_processes: # if the user specifies whether to override that's fine if set_default in [True, False]: _set_default = set_default # otherwise we may need to ask else: default_profile = default_profiles.get(process, '') if default_profile: _set_default = click.confirm("The default profile for the '{}' process is set to '{}': " "do you want to set the newly created '{}' as the new default? (can be reverted later)" .format(process, default_profile, profile_name)) # if there are no other default profiles, we don't need to ask else: _set_default = True if _set_default: set_default_profile(process, profile_name, force_rewrite=True) def _check_db_name(dbname, postgres): '''looks up if a database with the name exists, prompts for using or creating a differently named one''' create = True while create and postgres.db_exists(dbname): click.echo('database {} already exists!'.format(dbname)) if not click.confirm('Use it (make sure it is not used by another profile)?'): dbname = click.prompt('new name', type=str, default=dbname) else: create = False return dbname, create
[docs]class Run(VerdiCommand): """ Execute an AiiDA script """ def run(self, *args): from aiida.backends.utils import load_dbenv, is_dbenv_loaded if not is_dbenv_loaded(): load_dbenv() import argparse from aiida.cmdline.commands.shell import default_modules_list import aiida.orm.autogroup from aiida.orm.autogroup import Autogroup parser = argparse.ArgumentParser( prog=self.get_full_command_name(), description='Execute an AiiDA script.') parser.add_argument('-g', '--group', type=bool, default=True, help='Enables the autogrouping, default = True') parser.add_argument('-n', '--groupname', type=str, default=None, help='Specify the name of the auto group') # parser.add_argument('-o','--grouponly', type=str, nargs='+', default=['all'], # help='Limit the grouping to specific classes (by default, all classes are grouped') parser.add_argument('-e', '--exclude', type=str, nargs='+', default=[], help=('Autogroup only specific calculation classes.' " Select them by their module name.") ) parser.add_argument('-E', '--excludesubclasses', type=str, nargs='+', default=[], help=('Autogroup only specific calculation classes.' " Select them by their module name.") ) parser.add_argument('-i', '--include', type=str, nargs='+', default=['all'], help=('Autogroup only specific data classes.' " Select them by their module name.") ) parser.add_argument('-I', '--includesubclasses', type=str, nargs='+', default=[], help=('Autogroup only specific code classes.' " Select them by their module name.") ) parser.add_argument('scriptname', metavar='ScriptName', type=str, help='The name of the script you want to execute') parser.add_argument('new_args', metavar='ARGS', nargs=argparse.REMAINDER, type=str, help='Further parameters to pass to the script') parsed_args = parser.parse_args(args) # Prepare the environment for the script to be run globals_dict = { '__builtins__': globals()['__builtins__'], '__name__': '__main__', '__file__': parsed_args.scriptname, '__doc__': None, '__package__': None} ## dynamically load modules (the same of verdi shell) - but in ## globals_dict, not in the current environment for app_mod, model_name, alias in default_modules_list: globals_dict["{}".format(alias)] = getattr( __import__(app_mod, {}, {}, model_name), model_name) if parsed_args.group: automatic_group_name = parsed_args.groupname if automatic_group_name is None: import datetime now = datetime.datetime.now() automatic_group_name = "Verdi autogroup on " + now.strftime( "%Y-%m-%d %H:%M:%S") aiida_verdilib_autogroup = Autogroup() aiida_verdilib_autogroup.set_exclude(parsed_args.exclude) aiida_verdilib_autogroup.set_include(parsed_args.include) aiida_verdilib_autogroup.set_exclude_with_subclasses( parsed_args.excludesubclasses) aiida_verdilib_autogroup.set_include_with_subclasses( parsed_args.includesubclasses) aiida_verdilib_autogroup.set_group_name(automatic_group_name) ## Note: this is also set in the exec environment! ## This is the intended behavior aiida.orm.autogroup.current_autogroup = aiida_verdilib_autogroup try: f = open(parsed_args.scriptname) except IOError: print >> sys.stderr, "{}: Unable to load file '{}'".format( self.get_full_command_name(), parsed_args.scriptname) sys.exit(1) else: try: # Must add also argv[0] new_argv = [parsed_args.scriptname] + parsed_args.new_args with update_environment(new_argv=new_argv): # Add local folder to sys.path sys.path.insert(0, os.path.abspath(os.curdir)) # Pass only globals_dict exec (f, globals_dict) # print sys.argv except SystemExit as e: ## Script called sys.exit() # print sys.argv, "(sys.exit {})".format(e.message) ## Note: remember to re-raise, the exception to have ## the error code properly returned at the end! raise finally: f.close()
######################################################################## # HERE ENDS THE COMMAND FUNCTION LIST ######################################################################## # From here on: utility functions
[docs]def get_listparams(): """ Return a string with the list of parameters, to be printed The advantage of this function is that the calling routine can choose to print it on stdout or stderr, depending on the needs. """ max_length = max(len(i) for i in short_doc.keys()) name_desc = [(cmd.ljust(max_length + 2), desc.strip()) for cmd, desc in short_doc.iteritems()] name_desc = sorted(name_desc) return ("List of the most relevant available commands:" + os.linesep + os.linesep.join([" * {} {}".format(name, desc) for name, desc in name_desc]))
[docs]def get_command_suggestion(command): """ A function that prints on stderr a list of similar commands """ import difflib similar_cmds = difflib.get_close_matches(command, short_doc.keys()) if similar_cmds: print >> sys.stderr, "" print >> sys.stderr, "Did you mean this?" print >> sys.stderr, "\n".join([" {}".format(i) for i in similar_cmds])
def print_usage(execname): print >> sys.stderr, ("Usage: {} [--profile=PROFILENAME|-p PROFILENAME] " "COMMAND [<args>]".format(execname)) print >> sys.stderr, "" print >> sys.stderr, get_listparams() print >> sys.stderr, "See '{} help' for more help.".format(execname)
[docs]def exec_from_cmdline(argv): """ The main function to be called. Pass as parameter the sys.argv. """ ### This piece of code takes care of creating a list of valid ### commands and of their docstrings for dynamic management of ### the code. ### It defines a few global variables global execname global list_commands global short_doc global long_doc # import itself from aiida.cmdline import verdilib import inspect # List of command names that should be hidden or not completed. hidden_commands = ['completion', 'completioncommand', 'listparams'] # Retrieve the list of commands verdilib_namespace = verdilib.__dict__ list_commands = { verdi_subcmd.get_command_name(): verdi_subcmd for verdi_subcmd in verdilib_namespace.itervalues() if inspect.isclass(verdi_subcmd) and not verdi_subcmd == VerdiCommand and issubclass(verdi_subcmd, VerdiCommand) and not verdi_subcmd.__name__.startswith('_') and not verdi_subcmd._abstract } # Retrieve the list of docstrings, managing correctly the # case of empty docstrings. Each value is a list of lines raw_docstrings = {k: (v.__doc__ if v.__doc__ else "").splitlines() for k, v in list_commands.iteritems()} short_doc = {} long_doc = {} for k, cmd in list_commands.iteritems(): if k in hidden_commands: continue # assemble help string # first from python docstring if cmd.__doc__: help_msg = cmd.__doc__ else: help_msg = "" # if command has parser written with the 'click' module # we also add a dynamic help string documenting the options # Note: to enable this for a command, simply add a static # _ctx method (see e.g. Quicksetup) if hasattr(cmd, '_ctx'): help_msg += "\n" # resilient_parsing suppresses interactive prompts help_msg += cmd._ctx(args=[], resilient_parsing=True).get_help() help_msg = help_msg.split('\n') # need list of lines lines = [l.strip() for l in help_msg] empty_lines = [bool(l) for l in lines] try: first_idx = empty_lines.index(True) # The first non-empty line except ValueError: # All False short_doc[k] = "No description available" long_doc[k] = "" continue short_doc[k] = lines[first_idx] long_doc[k] = os.linesep.join(lines[first_idx + 1:]) execname = os.path.basename(argv[0]) try: profile, command_position = parse_profile(argv) except ProfileParsingException as err: print_usage(execname) sys.exit(1) # We now set the internal variable, if needed if profile is not None: settings_profile.AIIDADB_PROFILE = profile # I set the process to verdi settings_profile.CURRENT_AIIDADB_PROCESS = "verdi" # Finally, we parse the commands and their options try: command = argv[command_position] except IndexError: print_usage(execname) sys.exit(1) try: if command in list_commands: CommandClass = list_commands[command]() CommandClass.run(*argv[command_position + 1:]) else: print >> sys.stderr, ("{}: '{}' is not a valid command. " "See '{} help' for more help.".format( execname, command, execname)) get_command_suggestion(command) sys.exit(1) except ProfileConfigurationError as err: print >> sys.stderr, "The profile specified is not valid!" print >> sys.stderr, err.message sys.exit(1)
def run(): try: aiida.cmdline.verdilib.exec_from_cmdline(sys.argv) except KeyboardInterrupt: pass # print "CTRL+C caught, exiting from verdi..." except EOFError: pass # print "CTRL+D caught, exiting from verdi..."