Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Nojump #4031

Merged
merged 60 commits into from
Mar 28, 2023
Merged

Nojump #4031

Show file tree
Hide file tree
Changes from 13 commits
Commits
Show all changes
60 commits
Select commit Hold shift + click to select a range
e7f6b6f
First attempt at a nojump transformation
jvermaas Feb 19, 2023
96f43c3
This works now, matching the output from VMD for orthogonal trajectories
jvermaas Feb 19, 2023
14b724e
Add self to AUTHORS, add to CHANGELOG, make style happier.
jvermaas Feb 19, 2023
8a0c60d
This implementation works for non-orthogonal boxes, and is simpler th…
jvermaas Feb 20, 2023
2eb56bb
Merge branch 'nojump' of github.com:jvermaas/mdanalysis into nojump
jvermaas Feb 20, 2023
d16bbda
Added a test
jvermaas Feb 20, 2023
e05ecfd
Updates for Linter
jvermaas Feb 20, 2023
701cb10
How about now linter?
jvermaas Feb 20, 2023
5a4bd77
Whitespace fixes
jvermaas Feb 20, 2023
88f4395
Copy the right value
jvermaas Feb 20, 2023
081c127
Nope, made a mistake with the other one too
jvermaas Feb 20, 2023
dc06052
Add references, make darker happy.
jvermaas Feb 20, 2023
65c95c0
Spellcheck, actually define due and Doi
jvermaas Feb 20, 2023
5acec8a
Update test_nojump.py with a much nicer synthetic example
jvermaas Feb 21, 2023
5b2f067
Add to the docs
jvermaas Feb 21, 2023
f0d57ce
Add a check that we don't go backwards
jvermaas Feb 21, 2023
146cc07
Automake the nojump documentation
jvermaas Feb 21, 2023
c8263f1
Test every third value
jvermaas Feb 21, 2023
84527c8
Nojump
jvermaas Feb 21, 2023
635d15a
Test values in more places
jvermaas Feb 21, 2023
662a316
Actually add constant velocity test
jvermaas Feb 21, 2023
1ee4947
Actually provide a dimension
jvermaas Feb 21, 2023
e5a10e6
Trajectories can be transformed, universes cannot be
jvermaas Feb 21, 2023
112cb7d
Hopefully the last issue?
jvermaas Feb 21, 2023
43f3d0c
Fix variables
jvermaas Feb 21, 2023
232be2a
add wrap to the test
jvermaas Feb 21, 2023
f3e18a2
Ok, I think I was supposed to use Black to do the code reformatting t…
jvermaas Feb 21, 2023
79927ca
Black did something dumb I think
jvermaas Feb 21, 2023
08477ee
I forgot that wrapping in negative space can lead to problems
jvermaas Feb 21, 2023
64a4b1d
Bump up the tolerance a bit.
jvermaas Feb 21, 2023
f1b3588
Testing lint fix
jvermaas Feb 21, 2023
0eccb0a
Added some more tests to cover the branches.
jvermaas Feb 23, 2023
42c6112
Fix the tests again
jvermaas Feb 23, 2023
f4904c6
I'm definitely not used to how MDAnalysis does things.
jvermaas Feb 23, 2023
a3b1a21
Apparently I'm not covering a branch.
jvermaas Feb 23, 2023
3c2a90b
Make the tests better, also give the user a better error message if t…
jvermaas Feb 24, 2023
22953e0
I CANNOT trigger the warnings.
jvermaas Feb 24, 2023
faf8db9
I am now testing the warnings via a loaded file path
jvermaas Feb 24, 2023
b8afc96
MDAnalysis->mda:
jvermaas Feb 24, 2023
903cb29
Ok, misplaced an import
jvermaas Feb 24, 2023
535132e
No longer out of bounds
jvermaas Feb 24, 2023
e0e2b18
Catch busted Linverse during initialization.
jvermaas Feb 24, 2023
eec1758
Add an initial dimension, which hopefully gets ignored for subsequent…
jvermaas Feb 24, 2023
0e64312
Add text description for what the intended difference between
jvermaas Feb 24, 2023
be901c3
Separating no dimensions from invalid dimensions tests.
jvermaas Feb 26, 2023
5540da2
Merge branch 'nojump' of github.com:jvermaas/mdanalysis into nojump
jvermaas Feb 26, 2023
54b5651
Add some text explaining the notinvertible transform
jvermaas Feb 26, 2023
a484bec
Apply zero dimensions
jvermaas Feb 26, 2023
3ffdc27
Whoops. Actually make something non-invertible
jvermaas Mar 5, 2023
ff881fa
Merge branch 'develop' into nojump
jvermaas Mar 5, 2023
39183f0
Merge branch 'develop' into nojump
jvermaas Mar 15, 2023
47b0e39
Update __init__.py
jvermaas Mar 16, 2023
89d9eb7
Revert "Update __init__.py"
jvermaas Mar 16, 2023
997b221
Need to pass at least one non-zero dimension to trigger a LinAlgError
jvermaas Mar 26, 2023
70d23ba
Used mdanalysis.copy inside a test
jvermaas Mar 27, 2023
e40ee0f
Merge branch 'develop' into nojump
jvermaas Mar 27, 2023
1351bc4
Merge branch 'develop' into nojump
orbeckst Mar 28, 2023
7ae4e29
Update package/MDAnalysis/transformations/nojump.py
jvermaas Mar 28, 2023
49f5c01
Update package/MDAnalysis/transformations/nojump.py
jvermaas Mar 28, 2023
95f08a4
Update nojump.py
jvermaas Mar 28, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions package/AUTHORS
Original file line number Diff line number Diff line change
Expand Up @@ -206,6 +206,7 @@ Chronological list of authors
- Christian Pfaendner
- Pratham Chauhan
- Meet Brijwani
- Josh Vermaas


External code
Expand Down
2 changes: 2 additions & 0 deletions package/CHANGELOG
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,8 @@ Fixes
(Issue #3336)

Enhancements
* Add a nojump transformation, which unwraps trajectories so that particle
paths are continuous. (Issue #3703, PR #4031)
* Add distopia distance calculation library bindings as a selectable backend
for `calc_bonds` in `MDA.lib.distances`. (Issue #3783, PR #3914)
* AuxReaders are now pickle-able and copy-able (Issue #1785, PR #3887)
Expand Down
1 change: 1 addition & 0 deletions package/MDAnalysis/transformations/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -129,6 +129,7 @@ def wrapped(ts):
from .translate import translate, center_in_box
from .rotate import rotateby
from .positionaveraging import PositionAverager
from .nojump import NoJump
from .fit import fit_translation, fit_rot_trans
from .wrap import wrap, unwrap
from .boxdimensions import set_dimensions
127 changes: 127 additions & 0 deletions package/MDAnalysis/transformations/nojump.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,127 @@
# -*- Mode: python; tab-width: 4; indent-tabs-mode:nil; coding:utf-8 -*-
# vim: tabstop=4 expandtab shiftwidth=4 softtabstop=4
#
# MDAnalysis --- https://www.mdanalysis.org
# Copyright (c) 2006-2017 The MDAnalysis Development Team and contributors
# (see the file AUTHORS for the full list of names)
#
# Released under the GNU Public Licence, v2 or any higher version
#
# Please cite your use of MDAnalysis in published work:
#
# R. J. Gowers, M. Linke, J. Barnoud, T. J. E. Reddy, M. N. Melo, S. L. Seyler,
# D. L. Dotson, J. Domanski, S. Buchoux, I. M. Kenney, and O. Beckstein.
# MDAnalysis: A Python package for the rapid analysis of molecular dynamics
# simulations. In S. Benthall and S. Rostrup editors, Proceedings of the 15th
# Python in Science Conference, pages 102-109, Austin, TX, 2016. SciPy.
# doi: 10.25080/majora-629e541a-00e
#
# N. Michaud-Agrawal, E. J. Denning, T. B. Woolf, and O. Beckstein.
# MDAnalysis: A Toolkit for the Analysis of Molecular Dynamics Simulations.
# J. Comput. Chem. 32 (2011), 2319--2327, doi:10.1002/jcc.21787
#

"""\
No Jump Trajectory Unwrapping --- :mod:`MDAnalysis.transformations.nojump`
=======================================================================================

Unwraps the trajectory such that atoms never move more than half a periodic box length.
The algorithm used is based on :cite:p:`Kulke2022`.

.. autoclass:: NoJump

"""
import numpy as np
import warnings

from .base import TransformationBase
from ..due import due, Doi


class NoJump(TransformationBase):
orbeckst marked this conversation as resolved.
Show resolved Hide resolved
"""
Returns transformed coordinates for the given timestep so that an atom
does not move more than half the periodic box size, and will not jump
across the periodic boundary. The algorithm used is based on :cite:p:`Kulke2022`,
equation B6 for non-orthogonal systems.

Example
-------

To calculate diffusion, one of the first steps is to compute a trajectory
where the atoms do not cross the periodic boundary.

.. code-block:: python

transformation = NoJump()
u.trajectory.add_transformations(transformation)
for ts in u.trajectory:
print(ts.positions)

In this case, ``ts.positions`` will return the NoJump unwrapped trajectory.
To reverse the process, wrap the coordinates again.

Returns
-------
MDAnalysis.coordinates.timestep.Timestep

References
----------
.. bibliography::
:filter: False
:style: MDA

Kulke2022

"""

@due.dcite(Doi("10.1021/acs.jctc.2c00327"),
description="Works through the orthogonal case, "
"and proposes the non-orthogonal approach.", path=__name__)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I can't quite remember how the description shows up. The best thing for you would be to try it out https://docs.mdanalysis.org/stable/documentation_pages/references.html#citations-using-duecredit and see if you're happy with how your paper is presented to users.


def __init__(
self,
check_continuity=True,
max_threads=None,
# NoJump transforms are inherently unparallelizable, since
# it depends on the previous frame's unwrapped coordinates
parallelizable=False,
):
super().__init__(max_threads=max_threads, parallelizable=parallelizable)
self.prev = None
self.old_frame = 0
self.check_c = check_continuity

def _transform(self, ts):
if self.prev is None:
self.prev = ts.positions @ np.linalg.inv(ts.triclinic_dimensions)
self.old_frame = ts.frame
return ts
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

How does this work when I go through the trajectory a second time:

u.add_transformation(NoJump) 
[ts for ts in u.trajectory]          
[ts for ts in u.trajectory[2:]]     # once the trajectory resets, will prev point to the correct starting value?

I think there should be a test that it's still correct the second time around. Keeping state is tricky business and our readers' behavior is not always obvious (for instance, they should rewind when you start iteration).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It makes ZERO sense to unwrap an already unwrapped trajectory, and so ideally the transformation should only ever act once... I think. I could adopt something similar to what positionaveraging does, and try and remember what direction the steps are going in. However, in principle, unwrapping an already unwrapped trajectory is basically a no-op, since the round term will always be zero. I suppose I'd just need to enact a reset.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Note that the trajectory will not stay unwrapped. It’s not in memory, the coordinates are read from disk again. (Sorry if I misunderstood your comment… typing quickly before they really force me to switch to airplane mode)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Only if the user is using the MemoryReader then changes are permanent but that’s handled automatically with transformations.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm, ok. Now it'll throw a warning that you are going in the opposite direction (or with an unequal stride). When it encounters this, it will reset the unwrapping procedure.

if self.check_c and np.abs(self.old_frame - ts.frame) != 1:
orbeckst marked this conversation as resolved.
Show resolved Hide resolved
warnings.warn(
"NoJump transform is only accurate when positions"
"do not move by more than half a box length."
"Currently jumping between frames with a step of more than 1."
"This might be fine, but depending on the trajectory stride,"
"this might be inaccurate.",
Warning,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We typically use UserWarning instead of bare Warning.

(We have a few dedicated warning classes in https://docs.mdanalysis.org/stable/documentation_pages/exceptions.html but we try to not proliferate more)

)
# Remember that @ is a shorthand for matrix multiplication.
orbeckst marked this conversation as resolved.
Show resolved Hide resolved
# np.matmul(a, b) is equivalent to a @ b
L = ts.triclinic_dimensions
Linverse = np.linalg.inv(L)
# Convert into reduced coordinate space
fcurrent = ts.positions @ Linverse
fprev = self.prev # Previous unwrapped coordinates in reduced box coordinates.
# Calculate the new positions in reduced coordinate space (Equation B6 from
# 10.1021/acs.jctc.2c00327). As it turns out, the displacement term can
# be moved inside the round function in this coordinate space, as the
# difference between wrapped and unwrapped coordinates is an integer.
newpositions = fcurrent - np.round(fcurrent - fprev)
# Convert back into real space
ts.positions = newpositions @ L
# Set things we need to save for the next frame.
self.prev = newpositions # Note that this is in reduced coordinate space.
self.old_frame = ts.frame

return ts
11 changes: 11 additions & 0 deletions package/doc/sphinx/source/references.bib
Original file line number Diff line number Diff line change
Expand Up @@ -759,3 +759,14 @@ @incollection{Waveren2005
booktitle = {},
id = {transformations}
}

@article{Kulke2022,
title = {Reversible Unwrapping Algorithm for Constant-Pressure Molecular Dynamics Simulations},
author = {Kulke, Martin and Vermaas, Josh V.},
year = {2022},
journal = {Journal of Chemical Theory and Computation},
volume = {18},
number = {10},
pages = {6161--6171},
doi = {10.1021/acs.jctc.2c00327}
}
57 changes: 57 additions & 0 deletions testsuite/MDAnalysisTests/transformations/test_nojump.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
###

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

remove

import numpy as np
import pytest
from numpy.testing import assert_array_almost_equal
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We use assert_allclose nowadays but haven't updated everything.


import MDAnalysis as mda
from MDAnalysis.transformations import NoJump
from MDAnalysisTests import datafiles as data


@pytest.fixture()
def nojump_universes():
"""
Create the universe objects for the tests.
"""
u = mda.Universe(data.PSF_TRICLINIC, data.DCD_TRICLINIC)
transformation = NoJump()
u.trajectory.add_transformations(transformation)
return u


@pytest.fixture()
def nojump_universes_nocheck():
"""
Create the universe objects for the tests.
Do not check for continunity
"""
u = mda.Universe(data.PSF_TRICLINIC, data.DCD_TRICLINIC)
transformation = NoJump(check_continuity=False)
u.trajectory.add_transformations(transformation)
return u


def test_nojump_fwd(nojump_universes):
"""
Test if the nojump transform is returning the correct
values when iterating forwards over the sample trajectory.
"""
# These were determined based on determining what the TRICLINIC system
# would return on a validated version that worked on triclinic and
# orthogonal systems for a small water trajectory.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

mention VMD and fastpbc explicitly (include version numbers if you have!); we have other cases that we checked against a VMD implementation. We don't consider it shameful to give credit to other software ;-). More importantly, the more detail is provided for validation, the more confidence someone has in the correctness.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, the TRICLINIC system just was too ugly. The synthetic system is easier to understand.

# These are specific to atoms 166 and 362, which moved the most in the
# trajectory.
ref_matrix_fwd1 = np.asarray([-3.42261, 11.28495, -1.37211])
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What are the red values and how did you obtain them? Add a comment for provenance.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

My workflow for testing this was probably non-ideal. I basically made water orthogonal and non-orthogonal water boxes, checked that the results were equivalent to what I got from VMD/fastpbc. I then ran the algorithm on existing triclinic systems, and did some sanity checks. These are two atomic positions that happened to move the most. Comments are now included to highlight this.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@jvermaas May I suggest making a test case with a single particle that travels at a constant speed through the box? That way it should be easy to verify the correcness of the unwrapping. You can do this directly in MDAnalysis using

import MDAnalysis 
import numpy as np

N = 1
u = mda.Universe.empty(N, trajectory=True)
coordinates = np.empty((1000,  # number of frames
                        u.atoms.n_atoms,
                        3))
coordinates = ... # generate wrapped coordinates for a particle moving at a constant speed in the box. 
# I can help you with this if you need. 
u.load_new(coordinates, order='fac')

You can then make this universe a test fixture.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oooh... That is cleverer @hmacdope . I'll let you know how I make out. I'll probably need to look at how to add PBC information to a universe, but there is probably documentation somewhere.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am going to make this comment a blocking review @jvermaas, let me know if you need any help, feel free to just ping me.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I really like @hmacdope 's idea.

$$ \mathbf{r}(t) = \mathbf{v}_0 t + \mathbf{r}_0 $$

with some odd angle of the initial velocity and a smallish box and then wrap the straight shot trajectory with the wrap() transform.

Copy link
Member

@hmacdope hmacdope Feb 21, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

RE setting PBC the following should work to give you a 10x10x10 orthorhombic box using the dimensions argument.

import MDAnalysis 
import numpy as np

N = 1
u = mda.Universe.empty(N, trajectory=True)
coordinates = np.empty((1000,  # number of frames
                        u.atoms.n_atoms,
                        3))

u.load_new(coordinates, order='fac', in_memory=True, dimensions=[10,10,10, 90.0, 90.0,90.0])

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I scrapped the old benchmark and replaced it with a synthetic one. I think its still intelligible, and I can explain why the non-orthogonal number is what it is.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Instead of calling these variables "matrix", call them something like ref_last_pos_166 or something that makes clearer what they are.

ref_matrix_fwd2 = np.asarray([3.674243, -8.725193, -0.07884017])
size = (
nojump_universes.trajectory.ts.positions.shape[0],
nojump_universes.trajectory.ts.positions.shape[1],
len(nojump_universes.trajectory),
)
parr = np.empty(size)
for ts in nojump_universes.trajectory:
parr[..., ts.frame] = ts.positions.copy()
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think you can replace the explicit loop with

transformed_coordinates = u.trajectory.timeseries()

By default, it has shape (n_frames, n_atoms, 3) so you have to change the lines below.

# Atoms 166 and 362 happen to move alot.
assert_array_almost_equal(ref_matrix_fwd1, parr[166, :, -1], decimal=5)
assert_array_almost_equal(ref_matrix_fwd2, parr[362, :, -1], decimal=5)