What is AiiDA?🔗
AiiDA is an open-source workflow manager geared towards automating complex calculations. At it's core it is a generic framework which handles the submission of calculations, manages the data storage of calculation results, and enables complex error handling to name a few features.
AiiDA requires several other services to run such as a Postrgres database and a RabbitMQ service. Both of these can be found at LaunchIT. Before starting your AiiDA installation process you should initiate both of these services and keep the information readily available.
Getting started with AiiDA🔗
The original development of AiiDA developed around the idea of allowing a personal laptop act as the central repository. It would then be able to log into other computers/servers to submit jobs and retrieve after they had finished. With the advancedment of firewalls and security, especially at the lab, it is not as feasible to configure AiiDA with this type of installation. The following instructions will outline the total installation process on LC machines at the lab.
Cloning and basic AiiDA Setup🔗
There are two ways that you can install the core AiiDA package into your system. For either case, it is highly recommended to create a python or conda environment to keep your AiiDA installation separate to prevent breaking possible dependencies. This can be done with
python -m venv /path/to/environment/aiida source /path/to/environment/aiida/bin/activate
or
conda create -n aiida conda activate aiida
Once you have your environment initialized, you can install AiiDA through pip
pip install aiida-core
or download and and install from the AiiDA git repository.
git clone https://github.com/aiidateam/aiida-core.git cd aiida-core git checkout v2.6.2 # Decide if there is a more recent version you will use. pip install -e .
Anytime that you will be working with AiiDA you will need to activate this environment to gain access to the AiiDA commands. Activating your environment adds the name of your environment before the command prompt. It also changes the default binaries for python and pip to be those found in your environment.
Creating a profile🔗
The first thing you'll need to do is create a profile that will be associated with all the simulations launched through AiiDA. The profile will control which Postgres database and RabbitMQ service you will be connecting to. This can be used to separate projects if needed.
The main command to interact with AiiDA is `verdi`. This command should be available to you after installing the AiiDA package. You can list all possible commands with `verdi --help`.
There are several ways to setup a profile. The first is typing
verdi profile setup core.psql_dos
This will prompt you for relative information to set up your profile. A somewhat easier method would be to simply pass a yaml file that the command will read to create your profile.
verdi profile setup core.psql_dos --config profile.yaml
This file should look something like this:
non_interactive: y profile: <profile_name> email: <email> first_name: <First> last_name: <Last> institution: LLNL set_as_default: true use_rabbitmq: y database_engine: core.psql_dos database_hostname: <hostname> # Postgres LaunchIT database_port: <port> # Postgres LaunchIT database_name: <name> # Postgres LaunchIT database_username: <username> # Postgres LaunchIT database_password: <password> # Postgres LaunchIT repository_uri: file:///path/to/where/you/want/the/repository
The last line with `repository_uri` should be set to space with a large amount of storage. Your /usr/workspace can work for this. At some point, AiiDA will create a hidden folder called `.aiida`. The location of this can change but is either in your home folder or near your installation of aiida-core. The repository_uri can also be pointed to that location such that `~/.aiida/repository/main` would be the location where you replace ~ with the path. At this point you should be able to see that you have a profile created with `verdi profile list` which should return your chosen username.
AiiDA recently has changed so that you can configure your RabbitMQ server separately. At the time of writing this tutorial, the command to configure the RabbitMQ did not appear to be working correctly. To configure RabbitMQ with AiiDA, you can find a file named `config.json` inside of the `.aiida` folder. Find your user profile and edit the default values that AiiDA provides for the RabbitMQ. These values can be found from your LaunchIT instance. It should look something similar to the following example.
"process_control": { "backend": "core.rabbitmq", "config": { "broker_protocol": "amqps", "broker_username": "<username>", "broker_password": "<password>", "broker_host": "<hostname>", "broker_port": <port>, "broker_virtual_host": "<virtual_host>", "broker_parameters": { "no_verify_ssl": "1", "cafile": "/etc/pki/tls/cert.pem" } } }
These must be inserted at the same level as the "storage" value in the config.json file. Additionally, for LC the value `broker_parameters` must also be added to the file and can be seen above.
At this point you can test the status of the AiiDA installation by typing `verdi status`. The only thing that should not be initiated at this point is the daemon which interacts with the RabbitMQ server. This can be started with
verdi daemon start
By default it starts with a single daemon. You can increase the number by setting a number at the end of the command or by increasing or decreasing the current number with
verdi daemon incr 1 verdi daemon decr 1
Additionally, a nice with to interact with the python api is to use
verdi shell
which essentially initiates an ipython instance with many of the AiiDA modules pre-loaded.
Set up and configure a computer🔗
Set up computer🔗
A computer within the AiiDA context is where calculations will be ran. This can be any machine that you can ssh into and submit jobs. To give an example, we will create a computer for the Ruby server here at the lab. When setting up a computer. You will want to create a directory where all the simulations will be ran. Ideally, this would be some kind of scratch file system that has plenty of space for your simulations. For Ruby we have the Lustre file system which can be found at "/p/lustre1/<username>". I would recommend creating a folder in that space named AiiDA or something that will prompt you not to delete it in the future.
Similarly to the profile setup process, we can provide a yaml file to simplify the installation process. This can be done with
verdi computer setup --config ruby.yaml
where ruby.yaml is the file that will contain the following settings
label: ruby hostname: "ruby.llnl.gov" # Make sure you can ssh to this before installation by typing `ssh ruby.lln.gov` description: "Ruby server at LLNL" transport: core.ssh scheduler: "core.slurm" work_dir: "/p/lustre1/<username>/AiiDA" # You will need to confirm which lustre folder you are a part of. mpirun_command: "srun -n {tot_num_mpiprocs}" mpiprocs_per_machine: "56" prepend_text: "" append_text: " " shebang: "#!/bin/bash"
Once the initial step has been taken to setup the computer you will need to finish configuring the computer with
verdi computer configure core.ssh configure_ruby.yml
where configure_ruby.yaml contains
username: "<lc_username>" port: 22 look_for_keys: true key_filename: "/g/g1/<username>/.ssh/id_ecdsa" # Path to where your SSH key is at timeout: 60 allow_agent: true proxy_command: "" compress: true gss_auth: false gss_kex: false gss_deleg_creds: false gss_host: "ruby" load_system_host_keys: true key_policy: "RejectPolicy" use_login_shell: true safe_interval: 10.0
AiiDA will ask you to set a maximum memory per machine (or node in this case) but this does not play well with the slurm scheduler on LC. You can tell it to ignore this value by putting ! as the value. If you put an initial value then you need to modify it by doing the following.
verdi shell # Loads a python shell with AiiDA functions loaded. computer = load_computer("ruby") # Should be the name of the server you are modifying. computer.set_default_memory_per_machine(None) computer.get_default_memory_per_machine() # This should return nothing at this point.
Test computer🔗
Before testing the computer, you will probably need to download a known_hosts file which contains the information of all the different servers on LC. This should replace your current known_hosts file on LC which should be located in the ~/.ssh folder.
Another possible issue can come from spurious output that occurs as you log into and out of the server. There is a line in your ~/.profile.linux file that echoes 'logout' after every session and will cause issues with AiiDA. Go into the file and comment out the last line to make it look like
# Set up the shell environment: #trap "echo 'logout'" 0
To make sure this computer is configured correctly, execute and make sure all tests pass:
verdi computer test ruby
Make sure that all tests pass at this point. Some common issues include needing to modify your .bashrc file to allow for non-interactive sessions. One way to do this is to prepend the following text to your .bashrc file.
if [[ $- != *i* ]] ; then return fi
Configure code🔗
At this point AiiDA can connect to your database/RabbitMQ service and connect to the different servers to which you are wanting to submit calculations. The next step in the process is to configure a code executable so that AiiDA knows where it is located. For an example, we will show how INQ and VASP can be configured to run within AiiDA.
AiiDA works by relying on plugins to specify how a code will interact with the AiiDA framework. Many of the popular code packages already have a community that supports a plugin and can found at the AiiDA plugin registry.
Before installing either of these plugins, we can check what current calculation plugins are currently installed with something like:
verdi plugin list aiida.calculations Registered entry points for aiida.calculations: * arithmetic.add * core.transfer * templatereplacer Report: Pass the entry point as an argument to display detailed information
Install plugin for INQ🔗
To install the plugin for INQ, execute the following commands:
git clone https://github.com/LLNL/aiida-inq cd aiida-inq pip install -e .
After installing aiida-inq, the plugin list should be updated to show the new calculations that can be performed.
Set up INQ code🔗
To set up INQ with a script, create file inq_code.yml with the following content
label: inq description: "INQ code from LLNL" default_calc_job_plugin: "inq.inq" filepath_executable: "/bin/bash" computer: ruby use_double_quotes: false with_mpi: true prepend_text: "" append_text: ""
You might notice that the path to the executable has been set to "/bin/bash". The AiiDA implementation for INQ works by creating a bash script. Inside of that script it will make reference to the INQ binary. With that being the case, make sure that the INQ binary can be found in your $PATH. This can be done by appending the following line to your .bashrc file
export PATH="/path/to/inq/binary:$PATH"
To install the code within AiiDA, you can then execute the following command:
verdi code create core.code.installed --config inq_code.yml
To check if the code has correctly installed and is recognized by AiiDA, you can execute:
verdi code list
Run an INQ calculation🔗
Running a job from a script🔗
One of the easiest ways to launch a job within AiiDA is through simple python scripts. An example script, launch.py, is shown below
from aiida import load_profile from aiida.orm import load_code, Dict from aiida.plugins import CalculationFactory, DataFactory from aiida.engine import run,submit from ase.build import bulk # Initiate the default profile load_profile() # Get the calculator from AiiDA InqCalculation = CalculationFactory('inq.inq') # Find the code you will use for the calculation code = load_code('inq@ruby') # Create a structure StructureData = DataFactory('core.structure') atoms = bulk('Si', crystalstructure='diamond', a=5.43) atoms.positions structure = StructureData(ase=atoms) inputs = { 'code': code, 'structure': structure, 'parameters' : Dict(dict={ 'electrons': { 'cutoff': '35.0 Ha', 'extra-states': 3 }, 'kpoints': { 'gamma': '', 'insert': '-0.5 -0.5 -0.5 0.0' }, 'ground-state': { 'tolerance': 1e-8 }, 'run': { 'ground-state': '' } }), 'metadata': { #'dry_run': True, # If uncommented, will only create the files. #'store_provenance': False, # Will not store any of the provenance in the database. 'options': { 'resources': { 'tot_num_mpiprocs': 4 } } } } # Will show detailed results run(InqCalculation, **inputs) # Comment the previous line and use the following if you want to get the pk value to follow along with. #calc = submit(InqCalculation, **inputs) #print(f'Created calculation with PK={calc.pk}')
which can then be run via
verdi run launch.py
You can again learn the pk of this number by running `verdi process list` or `verdi process list -a` to show all calculations. Check the help documentation, `verdi process list --help`, to see all the possible options.
Install plugin for VASP🔗
In order to run VASP you will need to have a current license. If you are not sure if you have access you should reach out to your group or project leaders to see. To install the plugin for VASP, execute the following commands:
pip install aiida-vasp
This will install the latest stable version of aiida-vasp. After installing aiida-vasp, the plugin list should be updated to show the new calculations that can be performed.
Set up VASP code🔗
To set up VASP with a script, create file vasp_code.yml with the following content
label: vasp_std description: "VASP code Ruby server at LLNL" default_calc_job_plugin: "vasp.vasp" filepath_executable: "/path/to/vasp/binary" computer: ruby use_double_quotes: false with_mpi: true prepend_text: "" append_text: ""
To install the code within AiiDA, you can then execute the following command:
verdi code create core.code.installed --config vasp_code.yml
To check if the code has correctly installed and is recognized by AiiDA, you can execute:
verdi code list
Install VASP POTCAR files🔗
VASP comes with proprietary PAW pseudopotential files which are named POTCAR. Elements can have multiple pseudopotential files depending on the number of electrons that were included when creating the file. To upload these to your AiiDA instance you will need to upload the files. This can be done by providing a tar file with all the POTCAR files. The format should be the same as they are provided from the VASP software.
% verdi data vasp-potcar uploadfamily --path=/path/to/potpaw_PBE.54.tar --name=PBE.54 --description="PBE potentials version 54" POTCAR files found: 327. New files uploaded: 327, Added to Family: 327
The name that you provide here is how the pseudopotential family will be referenced by AiiDA later.
Run a VASP calculation🔗
Running a job from a script🔗
One of the easiest ways to launch a job within AiiDA is through simple python scripts. An example script, launch.py, is shown below
from aiida import load_profile from aiida.orm import load_code, load_group, Str, Group, Int from aiida.plugins import DataFactory, WorkflowFactory from aiida.common.extendeddicts import AttributeDict from ase.io import read from ase.build import bulk, sort from aiida.engine import submit, run # Initiate the default profile load_profile() # Initiate workchain and other inputs workchain = WorkflowFactory('vasp.relax') inputs = AttributeDict() settings = AttributeDict() dict_data = DataFactory('core.dict') kpoints_data = DataFactory('core.array.kpoints') Bool = DataFactory('core.bool') # Settings settings.parser_settings = { 'include_node': ['energies', 'trajectory'], 'include_quantity': ['forces', 'stress'], 'electronic_step_energies': True } inputs.settings = dict_data(dict=settings) # Find the code you will use for the calculation code = load_code('vasp_std@ruby') # This will change based on the computer you install on. # Structure information StructureData = DataFactory('core.structure') atoms = bulk('Si', crystalstructure='diamond', a=5.43) structure = StructureData(ase=atoms) inputs.structure = structure # KPOINTS kpoints = kpoints_data() kpoints.set_kpoints_mesh([3,3,3]) inputs.kpoints = kpoints # INCAR inputs.parameters = dict_data(dict={ 'incar': { 'algo': 'Conjugate', 'encut': 500, 'prec': 'ACCURATE', 'ediff': 1E-4, 'ispin': 2, 'magmom': [0]*len(atoms), 'lorbit': 11, 'ismear': 0, 'sigma': 0.1, 'gga': 'PS', 'kpar': 2, 'ncore': 14, 'nelm': 500 } }) inputs.converge = AttributeDict({'pwcutoff_samples': Int(15)}) # POTCAR information inputs.potential_family = Str('PBE.54') # Name you previously specified when uploading POTCAR inputs.potential_mapping = dict_data(dict={'Si': 'Si'}) # Submission options options = AttributeDict() options.account = 'bank_name' # Name of the bank on LC options.queue_name = 'pbatch' options.max_wallclock_seconds = 60 * 60 * 12 options.resources = {'num_machines': 1} # Number of nodes inputs.options = dict_data(dict=options) # Relax options relax = AttributeDict() relax.perform = Bool(True) # Select relaxation algorithm relax.algo = DataFactory('core.str')('cg') # Set force cutoff limit (EDIFFG, but no sign needed) relax.force_cutoff = DataFactory('core.float')(0.01) # Turn on relaxation of positions (strictly not needed as the default is on) # The three next parameters correspond to the well known ISIF=3 setting relax.positions = DataFactory('core.bool')(True) # Turn on relaxation of the cell shape (defaults to False) relax.shape = DataFactory('core.bool')(True) # Turn on relaxation of the volume (defaults to False) relax.volume = DataFactory('core.bool')(True) # Set maximum number of ionic steps relax.steps = DataFactory('core.int')(100) inputs.relax = relax # Label inputs.label = Str('Pu2O3 structure optimization') inputs.description = Str('Structure optimization of Pu2O3 without optimized spin states. Will be computed afterwards.') inputs.clean_workdir = False # Submit the workchain calc = run(workchain, **inputs) print(f'Launched geometry optimization with PK={calc.pk}')
This comprehensive script will launch a calculation of silicon. There are several parameters to consider when launching. The settings portion of the inputs contains the parser settings and tells AiiDA which sections to parse. This list is rather large and you should consult the documentation for aiida-vasp to see how to get a particular property. The options section contains the job submission details such as bank, queue, resources, and walltime. There are many options for specifying how to optimize the geometry of the structure. Most of the options have been listed here to show the full capabilities of the relax workflow.
The launch.py script can then be run via
verdi run launch.py
The pk of the launched job can be found by running `verdi process list` or `verdi process list -a` to show all calculations. Check the help documentation, `verdi process list --help`, to see all the possible options.
Checking job status in AiiDA🔗
If you run `verdi process list`, current calculation will show up at the bottom of the list, with its identifying “pk” number in the leftmost column:
verdi process list PK Created Process label Process State Process status ---- --------- --------------- --------------- ---------------- 3339 13m ago InqCalculation ⏹ Finished [0]
You can then take that number and run verdi process show <pk> to learn more about the process.
Property Value ----------- ------------------------------------------------------------- type InqTDDFTWorkChain state Finished [11] The process did not register a required output. pk 3339 uuid c9d7fde2-12bf-4901-9893-3954695ff78c label description ctime 2024-10-01 17:58:04.092209+00:00 mtime 2024-10-01 17:58:18.093927+00:00 Inputs PK Type ------------------ ---- ------------- gs inq code 1 InstalledCode structure 3331 StructureData parameters 3332 Dict clean_workdir 3333 Bool max_iterations 3334 Int tddft inq code 1 InstalledCode structure 3331 StructureData parameters 3335 Dict clean_workdir 3336 Bool max_iterations 3337 Int clean_workdir 3338 Bool structure 3331 StructureData Outputs PK Type ----------------- ---- ------------- output_parameters 3354 Dict output_structure 3353 StructureData Called PK Type ------------ ---- ---------------- Ground_State 3341 InqBaseWorkChain TDDFT 3348 InqBaseWorkChain Log messages --------------------------------------------- There are 3 log messages for this calculation Run 'verdi process report 3339' to see them
Troubleshooting a job🔗
Once you have the pk number of a job, you can learn more about the calculation and its outputs via
verdi process show <pk> verdi process report <pk> verdi calcjob gotocomputer <pk>
The last option takes you to a directory where you’ll see the outputs from the job and the script actually submitted to slurm to run the desired calculations.
If necessary, you can kill a process with verdi process kill <pk>.
Getting results🔗
To look at the results of the calculation you can either write a script or start a verdi shell instance. The following code will get the results of the previous calculation.
verdi shell node = load_node(3339) outputs = node.outputs.output_parameters.get_dict() print(outputs)
Conclusion🔗
Hopefully this brief tutorial is enough to get you started running calculations using AiiDA on the LC servers at LLNL. For further documentation we highly suggest going over the documentation and tutorials that AiiDA has on their website.