Fitting phonon-accurate potentials#

This tutorial will show you how to control the MLIP fit settings with the autoplex workflow. The choice of the correct fit setup and hyperparameter settings has a significant influence on the final result. Please note that the fitting might need nodes with very large memory requirements (1 TB) in some cases.

General settings#

There are two categories of fit settings that you can change. The first type concerns the general fit setup, that will affect the fit regardless of the chosen MLIP method, and e.g. changes database specific settings (like the split-up into training and test data). The other type of settings influences the MLIP specific setup like e.g. the choice of hyperparameters.

In case of the general settings, you can pass the MLIP model you want to use with the ml_models parameter list. You can set the maximum force threshold force_max for filtering the data (“distillation”) in the MLIP fit preprocess step. In principle, the distillation step can be turned off by passing distillation=False, but it is strongly advised to filter out too high force data points. The hyperparameters and further parameters can be passed in the make call (see below) using fit_kwargs_list.

complete_flow = CompleteDFTvsMLBenchmarkWorkflow(
    ml_models=["GAP", "MACE"], ..., 
    apply_data_preprocessing=True, 
    f_max=40.0, split_ratio=0.4,
    num_processes_fit=48,
    benchmark_kwargs={"relax_maker_kwargs": {"relax_cell": False, "relax_kwargs": ...}, "calculator_kwargs": {"device": "cpu"}}
).make(..., 
    fit_kwargs_list=[
        {"general": {"two_body": True, "three_body": False, "soap": False}},  # GAP parameters
        {"model": "MACE", "device": "cuda"}  # MACE parameters  
        # fit_kwargs_list has to have the same order as in ml_models
    ],
    ...  # put the other hyperparameter commands here as shown below
)

The MLIP model specific settings and hyperparameters setup varies from model to model and is demonstrated in the next sections. Also, atomate2-based MLPhononMaker settings can be changed via benchmark_kwargs as shown in the code snippet.

ℹ️ Note that autoplex provides the most comprehensive features for GAP, and more features for the other models will follow in future versions.

GAP#

There are several overall settings for the GAP fit that will change the mode in which autoplex runs. When hyper_para_loop is set to True, autoplex wil automatically iterate through a set of several hyperparameters (atomwise_regularization_list, soap_delta_list and n_sparse_list) and repeat the GAP fit for each combination. More information on the atom-wise regularization parameter can be found in J. Chem. Phys. 153, 044104 (2020) and a comprehensive list GAP hyperparameters can be found in the QUIP/GAP user guide. The other keywords to change autoplex’s mode are glue_xml (use glue.xml core potential instead of 2b/3b terms), regularization (use a sigma regularization) and separated (repeat the GAP fit for the combined database and each separated subset). The parameter atom_wise_regularization can turn the atom-wise regularization on and off, atomwise_regularization_parameter is the value that shall be set and force_min is the lower bound cutoff of forces taken into account for the atom-wise regularization or otherwise be replaced by the f_min value. auto_delta let’s you decide if you want to pass a fixed delta value for the 2b, 3b and SOAP terms or let autoplex automatically determine a suitable delta value based on the database’s energies.

complete_flow = CompleteDFTvsMLBenchmarkWorkflow(
    ml_models=["GAP"], ...,
    apply_data_preprocessing=True,
    atom_wise_regularization=True, 
    atomwise_regularization_parameter=0.1, 
    force_min=0.01, 
    auto_delta=False,
    hyper_para_loop=True, 
    atomwise_regularization_list=[0.01, 0.1], 
    soap_delta_list=[0.5, 1.0, 1.5], 
    n_sparse_list=[1000, 3000, 6000, 9000]
).make(..., 
    structure_list=[structure],
    mp_ids=["mpid"],
    benchmark_mp_ids=["mpid"],
    benchmark_structures=[structure],
    glue_xml=False,
    regularization=False,
    separated=False,
    fit_kwargs_list=[{
     "general": {"default_sigma": "{0.001 0.05 0.05 0.0}", {"two_body": True, "three_body": False,"soap": False},...},
     "twob": {"cutoff": 5.0,...},
     "threeb": {"cutoff": 3.25,...},
     "soap": {"delta": 1.0, "l_max": 12, "n_max": 10,...},
    }]
)

autoplex provides a JSON dict file containing default GAP fit settings in autoplex/fitting/common/mlip-phonon-defaults.json, that can be overwritten using the fit keyword arguments as demonstrated in the code snippet.

autoplex follows a certain convention for naming files and labelling the data (see autoplex/fitting/common/mlip-phonon-defaults.json).

  "general": {
    "at_file": "train.extxyz",
    "energy_parameter_name": "REF_energy",
    "force_parameter_name": "REF_forces",
    "virial_parameter_name": "REF_virial",
    "gp_file": "gap_file.xml"
  },

You can either adapt to the autoplex conventions or change by passing your preferred names and label to the fit keyword arguments.

ACE#

For fitting and validating ACE potentials, one needs to install julia as autoplex relies on ACEpotentials.jl which support fitting of linear ACE. Currently no python package exists for the same.

complete_flow = CompleteDFTvsMLBenchmarkWorkflow(
    ml_models=["J-ACE"], ..., apply_data_preprocessing=True,
).make(..., 
    structure_list=[structure],
    mp_ids=["mpid"],
    benchmark_mp_ids=["mpid"],
    benchmark_structures=[structure],
    fit_kwargs_list=[{
        "order": 3,
        "totaldegree": 6,
        "cutoff": 2.0,
        "solver": "BLR"
    }],
    ...)

The ACE fit hyperparameters can be passed in the make call with its distinct commands. Because there is no respective MLPhononMaker in atomate2 for J-ACE, the functionalities for autoplex are limited to the data generation and ML fit in this case.

Nequip#

The Nequip fit procedure can be controlled by fit hyperparameters in the make call.

complete_flow = CompleteDFTvsMLBenchmarkWorkflow(
    ml_models=["Nequip"], ..., apply_data_preprocessing=True,
).make(...,
    structure_list=[structure],
    mp_ids=["mpid"],
    benchmark_mp_ids=["mpid"],
    benchmark_structures=[structure],
    fit_kwargs_list=[{
        "r_max": 4.0,
        "num_layers": 4,
        "l_max": 2,
        "num_features": 32,
        "num_basis": 8,
        "invariant_layers": 2,
        "invariant_neurons": 64,
        "batch_size": 5,
        "learning_rate": 0.005,
        "max_epochs": 10000,
        "default_dtype": "float32",
        "device": "cuda"
    }],
    ...
)

M3GNet#

In a similar way, the M3GNet fit hyperparameters can be passed using make as well.

complete_flow = CompleteDFTvsMLBenchmarkWorkflow(
    ml_models=["M3GNet"], ..., apply_data_preprocessing=True,
).make(...,
    structure_list=[structure],
    mp_ids=["mpid"],
    benchmark_mp_ids=["mpid"],
    benchmark_structures=[structure],
    fit_kwargs_list=[{
        "cutoff": 5.0,
        "threebody_cutoff": 4.0,
        "batch_size": 10,
        "max_epochs": 1000,
        "include_stresses": True,
        "hidden_dim": 128,
        "num_units": 128,
        "max_l": 4,
        "max_n": 4,
        "device": "cuda",
        "test_equal_to_val": True
    }],
    ...,
    )

MACE#

Here again, you can pass the MACE fit hyperparameters to make.

complete_flow = CompleteDFTvsMLBenchmarkWorkflow(
    ml_models=["MACE"], ..., apply_data_preprocessing=True,
).make(...,
    structure_list=[structure],
    mp_ids=["mpid"],
    benchmark_mp_ids=["mpid"],
    benchmark_structures=[structure],
    fit_kwargs_list=[{
        "model": "MACE",
        "config_type_weights": '{"Default": 1.0}',
        "hidden_irreps": "128x0e + 128x1o",
        "r_max": 5.0,
        "batch_size": 10,
        "max_num_epochs": 1500,
        "start_swa": 1200,
        "ema_decay": 0.99,
        "correlation": 3,
        "loss": "huber",
        "default_dtype": "float32",
        "device": "cuda"
    }],
    ...
)

Finetuning MACE-MP-0#

It is also possible to finetune MACE-MP-0. To do so, you need to install MACE-torch 0.3.7. Currently, this can only be done by cloning the git-repo and installing it from there: ACEsuit/mace. We currently install the main branch from there automatically within autoplex.

It is now important that you switch off the default settings for the fitting procedure (use_defaults_fitting=False). Please be careful with performing very low-data finetuning. Currently, we use a stratified split for splitting the data into train and test data, i.e. there will be at least one data point from the dataset including single displaced cells and one rattled structure.

The following workflow CompleteDFTvsMLBenchmarkWorkflowMPSettings uses Materials Project default settings slightly adapted to phonon runs (more accurate convergence, ALGO=Normal). It can also be used without finetuning option. To finetune optimally, please adapt the MACE fitting parameters yourself.

complete_workflow_mace = CompleteDFTvsMLBenchmarkWorkflowMPSettings(
        ml_models=["MACE"],
        volume_custom_scale_factors=[0.95,1.00,1.05], rattle_type=0, distort_type=0,
        apply_data_preprocessing=True, use_defaults_fitting=False,
        ...
    ).make(
        structure_list=[structure],
        mp_ids=["mpid"],
        benchmark_mp_ids=["mpid"],
        benchmark_structures=[structure],
        fit_kwargs_list=[{
            "model": "MACE",
            "name": "MACE_final",
            "foundation_model": "large",
            "multiheads_finetuning": False,
            "r_max": 6,
            "loss": "huber",
            "energy_weight": 1000.0,
            "forces_weight": 1000.0,
            "stress_weight": 1.0,
            "compute_stress": True,
            "E0s": "average",
            "scaling": "rms_forces_scaling",
            "batch_size": 1,
            "max_num_epochs": 200,
            "ema": True,
            "ema_decay": 0.99,
            "amsgrad": True,
            "default_dtype": "float64",
            "restart_latest": True,
            "lr": 0.0001,
            "patience": 20,
            "device": "cpu",
            "save_cpu": True,
            "seed": 3
        }],
    )

If you do not have internet access on the cluster, please make sure that you have downloaded and deposited the model that you want to finetune on the cluster beforehand. Instead of foundation_model="large", you can then simply set foundation_model="full_path_on_the_cluster"

Example script for autoplex workflow using GAP to fit and benchmark a Si database#

The following code snippet will demonstrate, how you can submit an autoplex workflow for an automated SOAP-only GAP fit and DFT benchmark for a Si allotrope database. The GAP fit parameters are taken from J. Chem. Phys. 153, 044104 (2020). In this example we will also use hyper_para_loop=True to loop through a set of given GAP fit convergence parameter and hyperparameters set as provided by the lists atomwise_regularization_list, soap_delta_list and n_sparse_list. In this example script, we are using jobflow_remote to submit the jobs to a remote cluster.

from jobflow_remote import submit_flow
from autoplex.auto.phonons.flows import CompleteDFTvsMLBenchmarkWorkflow
from mp_api.client import MPRester

mpr = MPRester(api_key='YOUR_MP_API_KEY')
struc_list = []
benchmark_structure_list = []
mpids = ["mp-149"]  # add all the Si structure mpids you are interested in
mpbenchmark = ["mp-149"]  # add all the Si structure mpids you are interested in
for mpid in mpids:
    struc = mpr.get_structure_by_material_id(mpid)
    struc_list.append(struc)
for mpbm in mpbenchmark:
    bm_struc = mpr.get_structure_by_material_id(mpbm)
    benchmark_structure_list.append(bm_struc)

autoplex_flow = CompleteDFTvsMLBenchmarkWorkflow(
    n_structures=50, symprec=0.1,
    volume_scale_factor_range=[0.95, 1.05], rattle_type=0, distort_type=0,
    hyper_para_loop=True, atomwise_regularization_list=[0.1, 0.01],
    apply_data_preprocessing=True,
    soap_delta_list=[0.5], n_sparse_list=[7000, 8000, 9000],
    split_ratio=0.33, regularization=False,
    separated=True, num_processes_fit=48,).make(
    structure_list=struc_list, mp_ids=mpids, benchmark_structures=benchmark_structure_list,
    benchmark_mp_ids=mpbenchmark,
    fit_kwargs_list=[{"soap": {"delta": 1.0, "l_max": 12, "n_max": 10,
                "atom_sigma": 0.5, "zeta": 4, "cutoff": 5.0,
                "cutoff_transition_width": 1.0,
                "central_weight": 1.0, "n_sparse": 9000, "f0": 0.0,
                "covariance_type": "dot_product",
                "sparse_method": "cur_points"},
       "general": {"two_body": False, "three_body": False, "soap": True,
                   "default_sigma": "{0.001 0.05 0.05 0.0}", "sparse_jitter": 1.0e-8, }}}],
)

autoplex_flow.name = "autoplex_wf"

resources = {...}

print(submit_flow(autoplex_flow, worker="autoplex_worker", resources=resources, project="autoplex"))

Running a MLIP fit only#

The following script shows an example of how you can run a sole GAP fit with autoplex using run_locally from jobflow for the job management.

#!/usr/bin/env python

from jobflow import SETTINGS
from autoplex.fitting.common.flows import MLIPFitMaker
import os
from jobflow import run_locally

os.environ["OMP_NUM_THREADS"] = "48" 
os.environ["OPENBLAS_NUM_THREADS"] = "1" 

os.chdir("/path/to/destination/directory")
store = SETTINGS.JOB_STORE
# connect to the job store
store.connect()


fit_input_dict = {
        "mp-id": {  # put a mp-id or another kind of data marker here
            "rattled_dir": [[
                (
                    "/path/to/randomized/supercell/structure/calculation"
                ),
                (
                    "/path/to/randomized/supercell/structure/calculation"
                ),
            ]],
            "phonon_dir": [[
                (
                    "/path/to/phonopy/supercell/structure/calculation"
                ),
            ]],
            "phonon_data": [],
        },
        "IsolatedAtom": {"iso_atoms_dir": [[
                (
                    "/path/to/isolated/atom/calculation"
                ),
            ]]
        }
    }
    
    
mlip_fit = MLIPFitMaker(
    mlip_type="GAP", ...,
    pre_xyz_files=["vasp_ref.extxyz"],
    pre_database_dir="/path/to/pre_database",
    auto_delta = True,
    glue_xml = False,
).make(
        species_list=["Li", "Cl"],
        fit_input=fit_input_dict,
        **{...}       
        )

run_locally(mlip_fit, create_folders=True, store=store)

Additional fit settings can again be passed using fit_kwargs or **{...}.

ℹ️ Note that in the current setup of autoplex, you need to pass a fit_input_dict to the MLIPFitMaker containing at least one entry for “rattled_dir”, “phonon_dir” and “isolated_atom” VASP calculations, otherwise the code will not finish successfully.

Is it possible to run the DFT calculations and the MLIP fitting step on different machines?#

Very often, we might have the situation that our GPU does not share a hard drive with the compute cluster where we perform the VASP runs. In such situations, it is convenient to split up the computations.

This can be done by e.g. using jobflow-remote and the following settings for VASP and fitting jobs. The local_worker is the local machine (e.g., a GPU without slurm queue).

from autoplex.auto.phonons.flows import CompleteDFTvsMLBenchmarkWorkflow, IterativeCompleteDFTvsMLBenchmarkWorkflow
from jobflow_remote import submit_flow, set_run_config
from atomate2.vasp.powerups import update_user_incar_settings
from atomate2.vasp.powerups import update_vasp_custodian_handlers

from atomate2.vasp.jobs.phonons import PhononDisplacementMaker
from atomate2.settings import Atomate2Settings

autoplex_flow = IterativeCompleteDFTvsMLBenchmarkWorkflow(max_iterations=3, rms_max=0.2,
                                                          complete_dft_vs_ml_benchmark_workflow_0=CompleteDFTvsMLBenchmarkWorkflow(
                                                              symprec=1e-3,
                                                              run_fits_on_different_cluster=True,
                                                              add_dft_phonon_struct=False,
                                                              path_to_hyperparameters="/local_machine/mlip-phonon-defaults.json",
                                                              apply_data_preprocessing=True,
                                                              add_dft_rattled_struct=True,
                                                              volume_custom_scale_factors=[1.0,1.0, 1.0],
                                                              rattle_type=0, distort_type=0,
                                                              rattle_std=0.1,  # maybe 0.1
                                                              benchmark_kwargs={"relax_maker_kwargs": {
                                                                  "relax_cell": False}},
                                                              supercell_settings={"min_length": 5,
                                                                                  "max_length": 15,
                                                                                  "min_atoms": 10,
                                                                                  "max_atoms": 300,
                                                                                  "fallback_min_length": 9},
                                                              # settings that worked with a GAP
                                                              split_ratio=0.33,
                                                              regularization=False,
                                                              separated=False,
                                                              num_processes_fit=48,
                                                              displacement_maker=phonon_displacement_maker,
                                                              phonon_bulk_relax_maker=phonon_bulk_relax_maker,
                                                              phonon_static_energy_maker=phonon_static_energy_maker,
                                                              rattled_bulk_relax_maker=phonon_bulk_relax_maker,
                                                              isolated_atom_maker=static_isolated_atom_maker),
                                                          complete_dft_vs_ml_benchmark_workflow_1=CompleteDFTvsMLBenchmarkWorkflow(
                                                              symprec=1e-3,
                                                              run_fits_on_different_cluster=True,
                                                              path_to_hyperparameters="/local_machine/mlip-phonon-defaults.json",
                                                              apply_data_preprocessing=True,
                                                              add_dft_phonon_struct=False,
                                                              add_dft_rattled_struct=True,
                                                              volume_custom_scale_factors=[1.0],
                                                              rattle_type=0, distort_type=0,
                                                              rattle_std=0.1,  
                                                              benchmark_kwargs={"relax_maker_kwargs": {
                                                                  "relax_cell": False}},
                                                              supercell_settings={"min_length": 5,
                                                                                  "max_length": 15,
                                                                                  "min_atoms": 10,
                                                                                  "max_atoms": 300,
                                                                                  "fallback_min_length": 9},
                                                              split_ratio=0.33,
                                                              regularization=False,
                                                              separated=False,
                                                              num_processes_fit=48,
                                                              displacement_maker=phonon_displacement_maker,
                                                              phonon_bulk_relax_maker=phonon_bulk_relax_maker,
                                                              phonon_static_energy_maker=phonon_static_energy_maker,
                                                              rattled_bulk_relax_maker=phonon_bulk_relax_maker,
                                                              isolated_atom_maker=static_isolated_atom_maker)).make(
    structure_list=structure_list, mp_ids=mpids, benchmark_structures=benchmark_structure_list,
    benchmark_mp_ids=mpbenchmark,
    rattle_seed=0,
    fit_kwargs_list=[{
        "soap": {"delta": 1.0, "l_max": 12, "n_max": 10,
                 "atom_sigma": 0.5, "zeta": 4, "cutoff": 5.0,
                 "cutoff_transition_width": 1.0,
                 "central_weight": 1.0, "n_sparse": 6000, "f0": 0.0,
                 "covariance_type": "dot_product",
                 "sparse_method": "cur_points"},
        "general": {"two_body": True, "three_body": False, "soap": True,
                    "default_sigma": "{0.001 0.05 0.05 0.0}", "sparse_jitter": 1.0e-8, }}]
)

resources = {"nodes": 1, "partition": "micro", "time": "00:55:00", "ntasks": 48, "qverbatim": "#SBATCH --get-user-env",
             "mail_user": "your_email@adress", "mail_type": "ALL", "account": "xxxxxx"}

resources_phon = {"nodes": 3, "partition": "micro", "time": "0:55:00", "ntasks": 144,
                  "qverbatim": "#SBATCH --get-user-env",
                  "mail_user": "your_email@adress", "mail_type": "ALL", "account": "xxxxxx"}

resources_ratt = {"nodes": 3, "partition": "micro", "time": "0:55:00", "ntasks": 144,
                  "qverbatim": "#SBATCH --get-user-env",
                  "mail_user": "your_email@adress", "mail_type": "ALL", "account": "xxxxxx"}

resources_mlip = {}
autoplex_flow = set_run_config(autoplex_flow, name_filter="dft static", resources=resources, worker="supermuc_worker")
autoplex_flow = set_run_config(autoplex_flow, name_filter="stat_iso_atom", resources=resources, worker="supermuc_worker")

autoplex_flow = set_run_config(autoplex_flow, name_filter="dft phonon static", resources=resources_phon, worker="supermuc_worker")
autoplex_flow = set_run_config(autoplex_flow, name_filter="static", resources=resources_phon, worker="supermuc_worker")

autoplex_flow = set_run_config(autoplex_flow, name_filter="dft rattle static", resources=resources_ratt, worker="supermuc_worker")
autoplex_flow = set_run_config(autoplex_flow, name_filter="tight relax", resources=resources, worker="supermuc_worker")
autoplex_flow = set_run_config(autoplex_flow, name_filter="dft tight relax", resources=resources, worker="supermuc_worker")

autoplex_flow = set_run_config(autoplex_flow, name_filter="machine_learning_fit", resources=resources_mlip, worker="local_worker")
autoplex_flow = set_run_config(autoplex_flow, name_filter="gap phonon static", resources=resources_mlip, worker="local_worker")
autoplex_flow = set_run_config(autoplex_flow, name_filter="Force field", resources=resources_mlip, worker="local_worker")
autoplex_flow = set_run_config(autoplex_flow, name_filter="data_preprocessing_for_fitting", resources=resources, worker="supermuc_worker")


autoplex_flow = update_user_incar_settings(autoplex_flow, {"NPAR": 4})

autoplex_flow = update_vasp_custodian_handlers(autoplex_flow, custom_handlers={})

autoplex_flow.name = "small Sn test, test without phonon2"

# submit the workflow to jobflow-remote
print(submit_flow(autoplex_flow, worker="local_worker", resources=resources_mlip, project="phonons_qha"))