.. DO NOT EDIT. .. THIS FILE WAS AUTOMATICALLY GENERATED BY SPHINX-GALLERY. .. TO MAKE CHANGES, EDIT THE SOURCE PYTHON FILE: .. "examples/atomistic/2-running-ase-md.py" .. LINE NUMBERS ARE GIVEN BELOW. .. only:: html .. note:: :class: sphx-glr-download-link-note :ref:`Go to the end ` to download the full example code. .. rst-class:: sphx-glr-example-title .. _sphx_glr_examples_atomistic_2-running-ase-md.py: .. _atomistic-tutorial-md: Running Molecular Dynamics with ASE =================================== .. py:currentmodule:: metatensor.torch.atomistic This tutorial shows how to use metatensor atomistic models to run Molecular Dynamics (MD) simulation using ASE. ASE is not the only way to run MD with metatensor models, but it is very easy to setup and use. .. TODO: once we have more MD engine, add link to their doc here .. GENERATED FROM PYTHON SOURCE LINES 15-47 .. code-block:: Python from typing import Dict, List, Optional # tools to run the simulation import ase.build import ase.md import ase.md.velocitydistribution import ase.units # tools for visualization import ase.visualize.plot import chemiscope import matplotlib.pyplot as plt # the usuals suspects import numpy as np import torch from metatensor.torch import Labels, TensorBlock, TensorMap from metatensor.torch.atomistic import ( MetatensorAtomisticModel, ModelCapabilities, ModelMetadata, ModelOutput, System, ) # Integration with ASE calculator for metatensor atomistic models from metatensor.torch.atomistic.ase_calculator import MetatensorCalculator .. GENERATED FROM PYTHON SOURCE LINES 49-73 The energy model ---------------- To run simulations, we'll need a model. For simplicity and keeping this tutorial short, we'll use a model for `Einstein's solid`_, where all atoms oscillate around their equilibrium position as harmonic oscillators, but don't interact with one another. The energy of each atoms is given by: .. math:: E_i = k * \left(\vec{r_i} - \vec{r_i}^0\right)^2, where :math:`k` is a harmonic force constant, :math:`\vec{r_i}` the position of atom :math:`i` and :math:`\vec{r_i}^0` the position of atom :math:`i` at equilibrium. The energy of the whole system is then .. math:: E = \sum_i E_i. .. _Einstein's solid: https://en.wikipedia.org/wiki/Einstein_solid Let's implement this model! You can check the :ref:`atomistic-tutorial-export` tutorial for more information on how to define your own custom models. .. GENERATED FROM PYTHON SOURCE LINES 74-127 .. code-block:: Python class HarmonicModel(torch.nn.Module): def __init__(self, force_constant: float, equilibrium_positions: torch.Tensor): """Create an ``HarmonicModel``. :param force_constant: force constant, in ``energy unit / (length unit)^2`` :param equilibrium_positions: torch tensor with shape ``n x 3``, containing the equilibrium positions of all atoms """ super().__init__() assert force_constant > 0 self.force_constant = force_constant self.equilibrium_positions = equilibrium_positions def forward( self, systems: List[System], outputs: Dict[str, ModelOutput], selected_atoms: Optional[Labels], ) -> Dict[str, TensorMap]: if list(outputs.keys()) != ["energy"]: raise ValueError( "this model can only compute 'energy', but `outputs` contains other " f"keys: {', '.join(outputs.keys())}" ) # we don't want to worry about selected_atoms yet if selected_atoms is not None: raise NotImplementedError("selected_atoms is not implemented") if outputs["energy"].per_atom: raise NotImplementedError("per atom energy is not implemented") # compute the energy for each system by adding together the energy for each atom energy = torch.zeros((len(systems), 1), dtype=systems[0].positions.dtype) for i, system in enumerate(systems): assert len(system) == self.equilibrium_positions.shape[0] r0 = self.equilibrium_positions energy[i] += torch.sum(self.force_constant * (system.positions - r0) ** 2) # add metadata to the output block = TensorBlock( values=energy, samples=Labels("system", torch.arange(len(systems)).reshape(-1, 1)), components=[], properties=Labels("energy", torch.tensor([[0]])), ) return { "energy": TensorMap(keys=Labels("_", torch.tensor([[0]])), blocks=[block]) } .. GENERATED FROM PYTHON SOURCE LINES 128-134 Initial simulation state ------------------------ Now that we have a model for the energy of our system, let's create some initial simulation state. We'll build a 3x3x3 super cell of diamond carbon. In practice, you could also read the initial state from a file using :py:func:`ase.io.read`. .. GENERATED FROM PYTHON SOURCE LINES 135-143 .. code-block:: Python primitive = ase.build.bulk(name="C", crystalstructure="diamond", a=3.567) ax = ase.visualize.plot.plot_atoms(primitive, radii=0.5) ax.set_xlabel("$\\AA$") ax.set_ylabel("$\\AA$") plt.show() .. image-sg:: /examples/atomistic/images/sphx_glr_2-running-ase-md_001.png :alt: 2 running ase md :srcset: /examples/atomistic/images/sphx_glr_2-running-ase-md_001.png :class: sphx-glr-single-img .. GENERATED FROM PYTHON SOURCE LINES 144-145 And now let's make a super cell with this .. GENERATED FROM PYTHON SOURCE LINES 145-154 .. code-block:: Python atoms = ase.build.make_supercell(primitive, 3 * np.eye(3)) ax = ase.visualize.plot.plot_atoms(atoms, radii=0.5) ax.set_xlabel("$\\AA$") ax.set_ylabel("$\\AA$") plt.show() .. image-sg:: /examples/atomistic/images/sphx_glr_2-running-ase-md_002.png :alt: 2 running ase md :srcset: /examples/atomistic/images/sphx_glr_2-running-ase-md_002.png :class: sphx-glr-single-img .. GENERATED FROM PYTHON SOURCE LINES 155-157 We'll also need to initialize the velocities of the atoms to match our simulation temperature. .. GENERATED FROM PYTHON SOURCE LINES 158-169 .. code-block:: Python # The atoms start with zero velocities, hence zero kinetic energy and zero temperature print("before:") print("kinetic energy =", atoms.get_kinetic_energy(), "eV") print("temperature =", atoms.get_temperature(), "K") ase.md.velocitydistribution.MaxwellBoltzmannDistribution(atoms, temperature_K=300) print("\nafter:") print("kinetic energy =", atoms.get_kinetic_energy(), "eV") print("temperature =", atoms.get_temperature(), "K") .. rst-class:: sphx-glr-script-out .. code-block:: none before: kinetic energy = 0.0 eV temperature = 0.0 K after: kinetic energy = 2.6605345137326406 eV temperature = 381.16335132183576 K .. GENERATED FROM PYTHON SOURCE LINES 170-178 Running the simulation ---------------------- The final step to run the simulation is to register our model as the energy calculator for these ``atoms``. This is the job of :py:class:`ase_calculator.MetatensorCalculator`, which takes either an instance of :py:class:`MetatensorAtomisticModel` or the path to a pre-exported model, and allow to use it to compute the energy, forces and stress acting on a system. .. GENERATED FROM PYTHON SOURCE LINES 179-204 .. code-block:: Python # define & wrap the model, using the initial positions as equilibrium positions model = HarmonicModel( force_constant=3.14159265358979323846, equilibrium_positions=torch.tensor(atoms.positions), ) capabilities = ModelCapabilities( outputs={ "energy": ModelOutput(quantity="energy", unit="eV", per_atom=False), }, atomic_types=[6], interaction_range=0.0, length_unit="Angstrom", supported_devices=["cpu"], dtype="float32", ) # we don't want to bother with model metadata, so we define it as empty metadata = ModelMetadata() wrapper = MetatensorAtomisticModel(model.eval(), metadata, capabilities) # Use the wrapped model as the calculator for these atoms atoms.calc = MetatensorCalculator(wrapper) .. GENERATED FROM PYTHON SOURCE LINES 205-209 We'll run the simulation in the constant volume/temperature thermodynamics ensemble (NVT or Canonical ensemble), using a Langevin thermostat for time integration. Please refer to the corresponding documentation (:py:class:`ase.md.langevin.Langevin`) for more information! .. GENERATED FROM PYTHON SOURCE LINES 210-238 .. code-block:: Python integrator = ase.md.Langevin( atoms, timestep=1.0 * ase.units.fs, temperature_K=300, friction=0.1 / ase.units.fs, ) # collect some data during the simulation trajectory = [] potential_energy = [] kinetic_energy = [] total_energy = [] temperature = [] for step in range(800): # run a single simulation step integrator.run(1) # collect data about the simulation potential_energy.append(atoms.get_potential_energy()) kinetic_energy.append(atoms.get_kinetic_energy()) total_energy.append(atoms.get_total_energy()) temperature.append(atoms.get_temperature()) if step % 10 == 0: trajectory.append(atoms.copy()) .. GENERATED FROM PYTHON SOURCE LINES 239-240 We can use `chemiscope `_ to visualize the trajectory .. GENERATED FROM PYTHON SOURCE LINES 241-244 .. code-block:: Python viewer_settings = {"bonds": False, "playbackDelay": 70} chemiscope.show(trajectory, mode="structure", settings={"structure": [viewer_settings]}) .. chemiscope:: _datasets/fig_2-running-ase-md_001.json.gz :mode: structure .. raw:: html


.. GENERATED FROM PYTHON SOURCE LINES 245-246 We can also look at the time evolution of some physical constants for our system: .. GENERATED FROM PYTHON SOURCE LINES 247-264 .. code-block:: Python fig, ax = plt.subplots(1, 2, figsize=(8, 5)) ax[0].plot(range(800), potential_energy, label="potential energy") ax[0].plot(range(800), kinetic_energy, label="kinetic energy") ax[0].plot(range(800), total_energy, label="total energy") ax[0].legend() ax[0].set_xlabel("t / fs") ax[0].set_ylabel("energy / eV") ax[1].plot(range(800), temperature) ax[1].set_xlabel("t / fs") ax[1].set_ylabel("temperature / K") fig.tight_layout() fig.show() .. image-sg:: /examples/atomistic/images/sphx_glr_2-running-ase-md_003.png :alt: 2 running ase md :srcset: /examples/atomistic/images/sphx_glr_2-running-ase-md_003.png :class: sphx-glr-single-img .. GENERATED FROM PYTHON SOURCE LINES 265-271 Using a pre-exported model -------------------------- As already mentioned, :py:class:`ase_calculator.MetatensorCalculator` also work with a pre-exported model, meaning you can also run simulations without defining or re-training a model: .. GENERATED FROM PYTHON SOURCE LINES 272-279 .. code-block:: Python wrapper.save("exported-model.pt") atoms.calc = MetatensorCalculator("exported-model.pt") print(atoms.get_potential_energy()) integrator.run(10) .. rst-class:: sphx-glr-script-out .. code-block:: none 1.6940622329711914 True .. rst-class:: sphx-glr-timing **Total running time of the script:** (0 minutes 2.655 seconds) .. _sphx_glr_download_examples_atomistic_2-running-ase-md.py: .. only:: html .. container:: sphx-glr-footer sphx-glr-footer-example .. container:: sphx-glr-download sphx-glr-download-jupyter :download:`Download Jupyter notebook: 2-running-ase-md.ipynb <2-running-ase-md.ipynb>` .. container:: sphx-glr-download sphx-glr-download-python :download:`Download Python source code: 2-running-ase-md.py <2-running-ase-md.py>` .. container:: sphx-glr-download sphx-glr-download-zip :download:`Download zipped: 2-running-ase-md.zip <2-running-ase-md.zip>` .. only:: html .. rst-class:: sphx-glr-signature `Gallery generated by Sphinx-Gallery `_