\(\renewcommand{\AA}{\text{Å}}\)

8.6.5. LAMMPS Python Tutorial


Overview

The lammps Python module is a wrapper class for the LAMMPS C language library interface API which is written using Python ctypes. The design choice of this wrapper class is to follow the C language API closely with only small changes related to Python specific requirements and to better accommodate object oriented programming.

In addition to this flat ctypes interface, the lammps wrapper class exposes a discoverable API that doesn’t require as much knowledge of the underlying C language library interface or LAMMPS C++ code implementation.

Finally, the API exposes some additional features for IPython integration into Jupyter notebooks, e.g. for embedded visualization output from dump style image.


Quick Start

System-wide or User Installation

Step 1: Building LAMMPS as a shared library

To use LAMMPS inside of Python it has to be compiled as shared library. This library is then loaded by the Python interface. In this example we enable the MOLECULE package and compile LAMMPS with PNG, JPEG and FFMPEG output support enabled.

mkdir $LAMMPS_DIR/build-shared
cd  $LAMMPS_DIR/build-shared

# MPI, PNG, Jpeg, FFMPEG are auto-detected
cmake ../cmake -DPKG_MOLECULE=yes -DPKG_PYTHON=on -DBUILD_SHARED_LIBS=yes
make

Step 2: Installing the LAMMPS Python package

Next install the LAMMPS Python package into your current Python installation with:

make install-python

This will create a so-called “wheel” and then install the LAMMPS Python module from that “wheel” into either into a system folder (provided the command is executed with root privileges) or into your personal Python module folder.

Note

Recompiling the shared library requires re-installing the Python package.

Installation inside of a virtual environment

You can use virtual environments to create a custom Python environment specifically tuned for your workflow.

Benefits of using a virtualenv

  • isolation of your system Python installation from your development installation

  • installation can happen in your user directory without root access (useful for HPC clusters)

  • installing packages through pip allows you to get newer versions of packages than e.g., through apt-get or yum package managers (and without root access)

  • you can even install specific old versions of a package if necessary

Prerequisite (e.g. on Ubuntu)

apt-get install python-venv

Creating a virtualenv with lammps installed

# create virtual envrionment named 'testing'
python3 -m venv $HOME/python/testing

# activate 'testing' environment
source $HOME/python/testing/bin/activate

Now configure and compile the LAMMPS shared library as outlined above. When using CMake and the shared library has already been build, you need to re-run CMake to update the location of the python executable to the location in the virtual environment with:

cmake . -DPython_EXECUTABLE=$(which python)

# install LAMMPS package in virtualenv
(testing) make install-python

# install other useful packages
(testing) pip install matplotlib jupyter mpi4py pandas

...

# return to original shell
(testing) deactivate

Creating a new lammps instance

To create a lammps object you need to first import the class from the lammps module. By using the default constructor, a new lammps instance is created.

from lammps import lammps
L = lammps()

See the LAMMPS Python documentation for how to customize the instance creation with optional arguments.


Commands

Sending a LAMMPS command with the library interface is done using the command method of the lammps object.

For instance, let’s take the following LAMMPS command:

region box block 0 10 0 5 -0.5 0.5

This command can be executed with the following Python code if L is a lammps instance:

L.command("region box block 0 10 0 5 -0.5 0.5")

For convenience, the lammps class also provides a command wrapper cmd that turns any LAMMPS command into a regular function call:

L.cmd.region("box block", 0, 10, 0, 5, -0.5, 0.5)

Note that each parameter is set as Python number literal. With the wrapper each command takes an arbitrary parameter list and transparently merges it to a single command string, separating individual parameters by white-space.

The benefit of this approach is avoiding redundant command calls and easier parameterization. With the command function each call needs to be assembled manually using formatted strings.

L.command(f"region box block {xlo} {xhi} {ylo} {yhi} {zlo} {zhi}")

The wrapper accepts parameters directly and will convert them automatically to a final command string.

L.cmd.region("box block", xlo, xhi, ylo, yhi, zlo, zhi)

Note

When running in IPython you can use Tab-completion after L.cmd. to see all available LAMMPS commands.


Accessing atom data

All per-atom properties that are part of the atom style in the current simulation can be accessed using the extract_atoms() method. This can be retrieved as ctypes objects or as NumPy arrays through the lammps.numpy module. Those represent the local atoms of the individual sub-domain for the current MPI process and may contain information for the local ghost atoms or not depending on the property. Both can be accessed as lists, but for the ctypes list object the size is not known and hast to be retrieved first to avoid out-of-bounds accesses.

nlocal = L.extract_setting("nlocal")
nall = L.extract_setting("nall")
print("Number of local atoms ", nlocal, "  Number of local and ghost atoms ", nall);

# access via ctypes directly
atom_id = L.extract_atom("id")
print("Atom IDs", atom_id[0:nlocal])

# access through numpy wrapper
atom_type = L.numpy.extract_atom("type")
print("Atom types", atom_type)

x = L.numpy.extract_atom("x")
v = L.numpy.extract_atom("v")
print("positions array shape", x.shape)
print("velocity array shape", v.shape)
# turn on communicating velocities to ghost atoms
L.cmd.comm_modify("vel", "yes")
v = L.numpy.extract_atom('v')
print("velocity array shape", v.shape)

Some properties can also be set from Python since internally the data of the C++ code is accessed directly:

# set position in 2D simulation
x[0] = (1.0, 0.0)

# set position in 3D simulation
x[0] = (1.0, 0.0, 1.)

Retrieving the values of thermodynamic data and variables

To access thermodynamic data from the last completed timestep, you can use the get_thermo() method, and to extract the value of (compatible) variables, you can use the extract_variable() method.

result = L.get_thermo("ke") # kinetic energy
result = L.get_thermo("pe") # potential energy

result = L.extract_variable("t") / 2.0

Error handling

We are using C++ exceptions in LAMMPS for errors and the C language library interface captures and records them. This allows checking whether errors have happened in Python during a call into LAMMPS and then re-throw the error as a Python exception. This way you can handle LAMMPS errors in the conventional way through the Python exception handling mechanism.

Warning

Capturing a LAMMPS exception in Python can still mean that the current LAMMPS process is in an illegal state and must be terminated. It is advised to save your data and terminate the Python instance as quickly as possible.

Using LAMMPS in IPython notebooks and Jupyter

If the LAMMPS Python package is installed for the same Python interpreter as IPython, you can use LAMMPS directly inside of an IPython notebook inside of Jupyter. Jupyter is a powerful integrated development environment (IDE) for many dynamic languages like Python, Julia and others, which operates inside of any web browser. Besides auto-completion and syntax highlighting it allows you to create formatted documents using Markup, mathematical formulas, graphics and animations intermixed with executable Python code. It is a great format for tutorials and showcasing your latest research.

To launch an instance of Jupyter simply run the following command inside your Python environment (this assumes you followed the Quick Start instructions):

jupyter notebook

Interactive Python Examples

Examples of IPython notebooks can be found in the python/examples/ipython subdirectory. To open these notebooks launch jupyter notebook inside this directory and navigate to one of them. If you compiled and installed a LAMMPS shared library with PNG, JPEG and FFMPEG support you should be able to rerun all of these notebooks.

Validating a dihedral potential

This example showcases how an IPython Notebook can be used to compare a simple LAMMPS simulation of a harmonic dihedral potential to its analytical solution. Four atoms are placed in the simulation and the dihedral potential is applied on them using a datafile. Then one of the atoms is rotated along the central axis by setting its position from Python, which changes the dihedral angle.

phi = [d \* math.pi / 180 for d in range(360)]

pos = [(1.0, math.cos(p), math.sin(p)) for p in phi]

x = L.numpy.extract_atom("x")

pe = []
for p in pos:
    x[3] = p
    L.cmd.run(0, "post", "no")
    pe.append(L.get_thermo("pe"))

By evaluating the potential energy for each position we can verify that trajectory with the analytical formula. To compare both solutions, we plot both trajectories over each other using matplotlib, which embeds the generated plot inside the IPython notebook.

_images/pylammps_dihedral.jpg

Running a Monte Carlo relaxation

This second example shows how to use the lammps Python interface to create a 2D Monte Carlo Relaxation simulation, computing and plotting energy terms and even embedding video output.

Initially, a 2D system is created in a state with minimal energy.

_images/pylammps_mc_minimum.jpg

It is then disordered by moving each atom by a random delta.

random.seed(27848)
deltaperturb = 0.2
x = L.numpy.extract_atom("x")
natoms = x.shape[0]

for i in range(natoms):
    dx = deltaperturb \* random.uniform(-1, 1)
    dy = deltaperturb \* random.uniform(-1, 1)
    x[i][0] += dx
    x[i][1] += dy

L.cmd.run(0, "post", "no")
_images/pylammps_mc_disordered.jpg

Finally, the Monte Carlo algorithm is implemented in Python. It continuously moves random atoms by a random delta and only accepts certain moves.

estart = L.get_thermo("pe")
elast = estart

naccept = 0
energies = [estart]

niterations = 3000
deltamove = 0.1
kT = 0.05

for i in range(niterations):
    x = L.numpy.extract_atom("x")
    natoms = x.shape[0]
    iatom = random.randrange(0, natoms)
    current_atom = x[iatom]

    x0 = current_atom[0]
    y0 = current_atom[1]

    dx = deltamove \* random.uniform(-1, 1)
    dy = deltamove \* random.uniform(-1, 1)

    current_atom[0] = x0 + dx
    current_atom[1] = y0 + dy

    L.cmd.run(1, "pre no post no")

    e = L.get_thermo("pe")
    energies.append(e)

    if e <= elast:
        naccept += 1
        elast = e
    elif random.random() <= math.exp(natoms\*(elast-e)/kT):
        naccept += 1
        elast = e
    else:
        current_atom[0] = x0
        current_atom[1] = y0

The energies of each iteration are collected in a Python list and finally plotted using matplotlib.

_images/pylammps_mc_energies_plot.jpg

The IPython notebook also shows how to use dump commands and embed video files inside of the IPython notebook.