Simulating the Nitrogen-Copper interaction#

In this tutorial, we discuss how covalent can be used to manage workflows to simulate the quantum interactions between Nitrogen molecules and a slab of Copper. The computations for the quantum interactions are performed using the Atomic Simulation Environment (ase) package.

The covalent library allows us to *orchestrate* and *execute* workflows comprised of subtasks. Once, the workflows have been dispatched for execution, we can query the *status* of the computations. As the subtasks are completed, the execution results can be *collected* using the covalent results manager. The workflow result, which will become available once all the subtasks have finished running, can be *collected* in a similar manner.

First, we install all the required packages.

[1]:
with open("./requirements.txt", "r") as file:
    for line in file:
        print(line.rstrip())

ase==3.22.1
ase-notebook==0.3.2
covalent
matplotlib==3.4.3
[2]:
# Install necessary packages
# !pip install -r ./requirements.txt

Next, we import the tools from the ase package required to perform computations for the quantum interactions. Note that ase.calculators.emt.EMT (imported below) is a low compute calculator, which, can be replaced by a high compute calculator and thus making the workflow more computationally expensive.

[3]:
import covalent as ct
from ase import Atoms
from ase.calculators.emt import EMT
from ase.constraints import FixAtoms
from ase.optimize import QuasiNewton
from ase.build import fcc111, add_adsorbate
from ase.io import read
from ase.io.trajectory import Trajectory
from ase_notebook import AseView, ViewConfig
from matplotlib import pyplot as plt

Our objective is to simulate the structural relaxation of the Nitrogen molecules and Copper slabs. The Nitrogen molecules are comprised of two Nitrogen atoms at a distance, \(d\), apart. The \(\text{N}_{2}\) molecules are placed at a height, \(h\), above the copper slab. In order to perform the structural relaxation computation, we fix the distance between the Nitrogen molecules and optimize the height using the quasi-Newtonian algorithm.

Now that we have outlined the purpose of the workflow, we can break it down into smaller subtasks (defined below).

  1. construct_cu_slab - Construct a copper slab.

  2. compute_system_energy - Computes the energy of the system in a particular configuration.

  3. construct_n_molecule - Construct a Nitrogen molecule with some spacing \(d\).

  4. get_relaxed_slab - Get relaxed slab structure where the height is optimized using the quasi-Newtonian method.

The subtasks are constructed below using the ct.electron decorator.

[4]:
@ct.electron
def construct_cu_slab(
    unit_cell=(4, 4, 2),
    vacuum=10.0,
):
    slab = fcc111("Cu", size=unit_cell, vacuum=vacuum)
    return slab


@ct.electron
def compute_system_energy(system):
    system.calc = EMT()
    return system.get_potential_energy()


@ct.electron
def construct_n_molecule(d=0):
    return Atoms("2N", positions=[(0.0, 0.0, 0.0), (0.0, 0.0, d)])


@ct.electron
def get_relaxed_slab(slab, molecule, height=1.85):
    slab.calc = EMT()
    add_adsorbate(slab, molecule, height, "ontop")
    constraint = FixAtoms(mask=[a.symbol != "N" for a in slab])
    slab.set_constraint(constraint)
    dyn = QuasiNewton(slab, trajectory="/tmp/N2Cu.traj", logfile="/tmp/temp")
    dyn.run(fmax=0.01)
    return slab

Next, we visualize the problem setup using tools from ase-notebook.

[5]:
N2 = construct_n_molecule(d=1.2)
slab = construct_cu_slab(unit_cell=(4, 4, 2), vacuum=10.0)
add_adsorbate(slab, N2, 1.2, "ontop")

config = ViewConfig(show_unit_cell=False)
ase_view = AseView(config)
ase_view.config.rotations = "-90x,50y,0z"
config.atom_lighten_by_depth = 0.7
ase_view.make_svg(
    slab,
    center_in_uc=True,
    repeat_uc=(2, 2, 1),
)

[5]:
../../../_images/tutorials_3_QuantumChemistry_nitrogen_copper_interaction_source_8_0.svg

Having defined the subtasks, we are now in a position to construct the workflow (compute_relaxed_structure_energy) using the ct.lattice decorator. The structural relaxation is performed by considering the hybridization interaction between Nitrogen and Copper, where, the quasi-Newton method is used to minimize the total kohn-sham energy of the system.

[6]:
@ct.lattice
def compute_energy(initial_height=3, distance=1.10):
    N2 = construct_n_molecule(d=distance)
    e_N2 = compute_system_energy(system=N2)

    slab = construct_cu_slab(unit_cell=(4, 4, 2), vacuum=10.0)
    e_slab = compute_system_energy(system=slab)

    relaxed_slab = get_relaxed_slab(slab=slab, molecule=N2, height=initial_height)
    e_relaxed_slab = compute_system_energy(system=relaxed_slab)
    final_result = e_slab + e_N2 - e_relaxed_slab

    return final_result

Once the workflow has been constructed, we can visualize the subtasks and their interdependencies using the built-in covalent visualization tool.

[7]:
compute_energy.draw(initial_height=3, distance=1.10)

One you run the above codeblock, you can check out the ui preview (default at localhost:48008/preview) and see the generated graph which will look like this:

[8]:
from IPython import display

display.Image("assets/compute_energy.png")

[8]:
../../../_images/tutorials_3_QuantumChemistry_nitrogen_copper_interaction_source_14_0.png

Next, we send off the workflow for execution via the dispatch method and by passing the input parameters. This method submits the job to the dispatcher server which then sends the subtasks to the appropriate executors.

[9]:
dispatch_id = ct.dispatch(compute_energy)(initial_height=3, distance=1.10)

After the workflow has been dispatched, we can now query the result using ct.get_result. Specifying the wait parameter to True will ensure that the result is returned only once all the computations have been completed. Lastly, the parameter results_dir needs to be specified if a non-default value (by default results_dir=./results) had been set in the lattice decorator.

[10]:
result = ct.get_result(dispatch_id=dispatch_id, wait=True)

print(
    f"Computation status = {result.status}\nEnergy of slab + nitrogen system = {result.result:.4f}"
)

Computation status = COMPLETED
Energy of slab + nitrogen system = 0.3242

Tip

For longer running jobs, we might want to set wait parameter to False, in which case, the result object is retrieved before the computations have finished executing. This allows us to query the *status* (and other relevant information such as execution start/end time) of each subtask using the get_node_result(node_id) method.

Furthermore, we can query the *result* of each subtask via get_all_node_outputs. The result of each node can be accessed with the key in the format ‘subtask_name(node_id)’. The node_id can be found by examining the workflow visualization.

[11]:
import pprint

pp = pprint.PrettyPrinter(indent=1)

pp.pprint(result.get_all_node_outputs())

{':parameter:(4, 4, 2)(4)': <covalent.TransportableObject object at 0x7fd7695eb520>,
 ':parameter:1.1(1)': <covalent.TransportableObject object at 0x7fd78a1321f0>,
 ':parameter:10.0(5)': <covalent.TransportableObject object at 0x7fd78a13c130>,
 ':parameter:3(8)': <covalent.TransportableObject object at 0x7fd78a13c5b0>,
 'compute_system_energy(2)': <covalent.TransportableObject object at 0x7fd78a132c10>,
 'compute_system_energy(6)': <covalent.TransportableObject object at 0x7fd78a13c3d0>,
 'compute_system_energy(9)': <covalent.TransportableObject object at 0x7fd78a13c850>,
 'compute_system_energy_+_compute_system_energy(10)': <covalent.TransportableObject object at 0x7fd78a13c9d0>,
 'compute_system_energy_+_compute_system_energy_-_compute_system_energy(11)': <covalent.TransportableObject object at 0x7fd78a13cb50>,
 'construct_cu_slab(3)': <covalent.TransportableObject object at 0x7fd78a132df0>,
 'construct_n_molecule(0)': <covalent.TransportableObject object at 0x7fd78a132f10>,
 'get_relaxed_slab(7)': <covalent.TransportableObject object at 0x7fd78a13c550>}

Further information on the subtask execution can be queried using the node_id param.

[12]:
pp.pprint(result.get_node_result(node_id=0))

{'end_time': datetime.datetime(2023, 1, 16, 18, 48, 1, 172504, tzinfo=datetime.timezone.utc),
 'error': None,
 'node_id': 0,
 'node_name': 'construct_n_molecule',
 'output': <covalent.TransportableObject object at 0x7fd78a132f10>,
 'start_time': datetime.datetime(2023, 1, 16, 18, 48, 1, 55157, tzinfo=datetime.timezone.utc),
 'status': Status(STATUS='COMPLETED'),
 'stderr': '',
 'stdout': '',
 'sublattice_result': None}

Making a new workflow based on (sub) workflows#

Having constructed a workflow to optimize the height for a given nitrogen molecule spacing, \(d\), we want to construct a workflow that optimizes the molecule spacing. Covalent allows us to construct workflows out of other workflows using the ct.electron function (shown below). Furthermore, Covalent automatically parallelizes the execution of each subworkflow.

[13]:
import numpy as np

optimize_height = ct.electron(compute_energy)


@ct.lattice
def vary_distance(distance):
    result = []
    for i in distance:
        result.append(optimize_height(initial_height=3, distance=i))
    return result

Alternatively, instead of constructing the sublattice via optimize_height = ct.electron(compute_energy), we could also have used the ct.electron decorator on top of the ct.lattice decorator when compute_energy was defined (syntax shown below).

@ct.electron
@ct.lattice
def compute_energy(**params):
    ...

We can visualize the new workflow using the syntax shown previously.

[14]:
import numpy as np

distance = np.linspace(1, 1.5, 3)

vary_distance.draw(distance=np.round(distance, 2))

This graph will look something like this:

[15]:
display.Image("assets/vary_distance.png")

[15]:
../../../_images/tutorials_3_QuantumChemistry_nitrogen_copper_interaction_source_31_0.png
[16]:
seps = 7
distance = np.linspace(1, 1.5, seps)
dispatch_id = ct.dispatch(vary_distance)(distance=distance)

One thing to note here is the value of seps which are the number of separations to be there in np.linspace. This directly translates to the number of sublattices that will be created and hence its dependent on the hardware upon which this experiment is being run. Thus, if you see that the experiment is taking a long time and is not finishing, then try again after lowering the value of seps.

[17]:
result = ct.get_result(dispatch_id, wait=True)
print(result.status)

COMPLETED

Lastly, we plot the workflow computation results of the system energy as a function of the nitrogen molecule spacing after performing structural relaxation of the copper slab and nitrogen molecule system.

[18]:
fig, ax = plt.subplots(1, 1, figsize=(6, 4), facecolor="w")
ax.plot(distance, result.result, marker="o")
ax.set_xlabel("Nitrogen molecule spacing")
ax.set_ylabel("Structural relaxation energy of Cu + ${N}_2$")
plt.tight_layout()
plt.show()

../../../_images/tutorials_3_QuantumChemistry_nitrogen_copper_interaction_source_36_0.png

Summary#

In this tutorial, we saw how to translate a computational simulation problem into a workflow comprised of subtasks using the covalent library with the electron and lattice decorator. The primary benefits of covalent are as follows:

  1. When lattices are executed, all the parallelizable sub-lattices (constructed via both electron and lattice decorator) are automatically executed in parallel

  2. Nested workflows can be constructed very easily making this a versatile tool for high performance computing.

  3. Lastly, covalent handles the serialization / deserialization of the workflows as the jobs are submitted to the server.