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

Implement iterative average structure (Issue#2039) #4524

Merged
merged 12 commits into from
Mar 29, 2024
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
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 @@ -236,6 +236,7 @@ Chronological list of authors
2024
- Aditya Keshari
- Philipp Stärk
- Leon Wehrhan

External code
-------------
Expand Down
4 changes: 3 additions & 1 deletion package/CHANGELOG
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ The rules for this file:
-------------------------------------------------------------------------------
??/??/?? IAlibay, HeetVekariya, marinegor, lilyminium, RMeli,
ljwoods2, aditya292002, pstaerk, PicoCentauri, BFedder,
tyler.je.reddy
tyler.je.reddy, leonwehrhan

* 2.8.0

Expand Down Expand Up @@ -55,6 +55,8 @@ Changes
equivalent assert_allclose(... rtol=0, atol=1.5e-{N}) (issue modernize
testing code #3743, PR Replaced numpy.testing.assert_almost_equal to
numpy.testing.assert_allclose #4438)
* Implement average structures with iterative algorithm from
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved
DOI 10.1021/acs.jpcb.7b11988. (Issue #2039, PR #4524)

Deprecations
* The MDAnalysis.analysis.waterdynamics module has been deprecated in favour
Expand Down
92 changes: 92 additions & 0 deletions package/MDAnalysis/analysis/align.py
Original file line number Diff line number Diff line change
Expand Up @@ -174,6 +174,7 @@
.. autoclass:: AlignTraj
.. autoclass:: AverageStructure
.. autofunction:: rotation_matrix
.. autofunction:: iterative_average


Helper functions
Expand Down Expand Up @@ -541,6 +542,97 @@ def alignto(mobile, reference, select=None, weights=None,
return old_rmsd, new_rmsd


def iterative_average(
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved
mobile, reference=None, select='all', weights=None, niter=100,
eps=1e-6, verbose=False, **kwargs
):
"""Calculate an optimal reference that is also the average structure
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved
after an RMSD alignment. The optimal reference is defined as average
structure of a trajectory, with the optimal reference used as input.
This function computes the optimal reference by using a starting
reference for the average structure, which is used as the reference
to calculate the average structure again. This is repeated until the
reference structure has converged. :footcite:p:`Linke2018`
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved

Parameters
----------
mobile : mda.Universe
Universe containing trajectory to be fitted to reference.
reference : mda.Universe (optional)
Universe containing the initial reference structure.
select : str or tuple or dict (optional)
Atom selection for fitting a substructue. Default is set to all.
Can be tuple or dict to define different selection strings for
mobile and target.
weights : str, array_like (optional)
Weights that can be used. If `None` use equal weights, if `'mass'`
use masses of ref as weights or give an array of arbitrary weights.
niter : int (optional)
Maximum number of iterations.
eps : float (optional)
RMSD distance at which reference and average are assumed to be
equal.
verbose : bool (optional)
Verbosity.
**kwargs : dict (optional)
AverageStructure kwargs.

Returns
-------
avg_struc : AverageStructure
AverageStructure result from the last iteration.

Example
-------

::
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved

import MDAnalysis as mda
from MDAnalysis.analysis import rms
from MDAnalysisTests.datafiles import PSF, DCD

u = mda.Universe(PSF, DCD)
av = rms.iterative_average(u, u, verbose=True)

averaged_universe = av.results.universe
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved

leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved
.. versionadded:: 2.8.0
"""
if not reference:
reference = mobile

select = rms.process_selection(select)
ref = mda.Merge(reference.select_atoms(*select['reference']))
sel_mobile = select['mobile'][0]

weights = get_weights(ref.atoms, weights)

drmsd = np.inf
for i in range(niter):
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved
# found a converged structure
if drmsd < eps:
break

avg_struc = AverageStructure(
mobile, reference=ref, select={
'mobile': sel_mobile, 'reference': 'all'
},
weights=weights, **kwargs
).run()
drmsd = rms.rmsd(ref.atoms.positions, avg_struc.results.positions,
weights=weights)
ref = avg_struc.results.universe

if verbose:
print(
"i = {}, rmsd-change = {:.5f}, ave-rmsd = {:.5f}".format(
i, drmsd, avg_struc.results.rmsd
)
)
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved

leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved
return avg_struc


class AlignTraj(AnalysisBase):
"""RMS-align trajectory to a reference structure using a selection.

Expand Down
2 changes: 2 additions & 0 deletions package/MDAnalysis/analysis/rms.py
Original file line number Diff line number Diff line change
Expand Up @@ -166,7 +166,9 @@
import logging
import warnings

import MDAnalysis as mda
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved
import MDAnalysis.lib.qcprot as qcp
import MDAnalysis.analysis.align as align
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved
from MDAnalysis.analysis.base import AnalysisBase
from MDAnalysis.exceptions import SelectionError, NoDataError
from MDAnalysis.lib.util import asiterable, iterable, get_weights
Expand Down
10 changes: 10 additions & 0 deletions package/doc/sphinx/source/references.bib
Original file line number Diff line number Diff line change
Expand Up @@ -770,3 +770,13 @@ @article{Kulke2022
pages = {6161--6171},
doi = {10.1021/acs.jctc.2c00327}
}

@article{Linke2018,
title = {Fully Anisotropic Rotational Diffusion Tensor from Molecular Dynamics Simulations},
author = {Linke, Max and Köfinger, Jürgen and Hummer, Gerhard},
year = {2018},
journal = {The Journal of Physical Chemistry B},
volume = {122},
number = {21},
pages = {5630--5639}
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved
}
58 changes: 58 additions & 0 deletions testsuite/MDAnalysisTests/analysis/test_align.py
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@
ALIGN_UNBOUND, PDB_helix)
from numpy.testing import (
assert_equal,
assert_almost_equal,
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved
assert_array_equal,
assert_allclose,
)
Expand Down Expand Up @@ -594,6 +595,63 @@ def test_sequence_alignment_deprecation(self, atomgroups):
align.sequence_alignment(mobile, reference)


class TestIterativeAverage(object):
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved
@pytest.fixture()
def mobile(self):
u = mda.Universe(PSF, DCD)
return u

@pytest.fixture()
def reference(self):
u = mda.Universe(PSF, DCD)
return u

def test_iterative_average(self, mobile, reference):
res = align.iterative_average(mobile, niter=1)
res = align.iterative_average(mobile, reference, niter=1, eps=0)
res = align.iterative_average(mobile, reference, eps=1e-5)
res = align.iterative_average(mobile, reference, select='bynum 1:10',
weights=np.ones(10))

res = align.iterative_average(mobile, reference, select='bynum 1:10',
niter=10, verbose=True)
assert_almost_equal(
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved
res.results.positions,
[
[11.93075595, 8.6729893, -10.49887605],
[12.60587898, 7.91673117, -10.73327464],
[12.45662411, 9.51900517, -10.35551193],
[11.27452274, 8.83003843, -11.2619057],
[11.25808119, 8.26794477, -9.23340715],
[12.02767222, 7.95332228, -8.57249317],
[10.54679871, 9.49505306, -8.61215292],
[9.99500556, 9.16624224, -7.75231192],
[9.83897407, 9.93134598, -9.29541129],
[11.45760169, 10.5857071, -8.13037669]
],
decimal=5,
)

res = align.iterative_average(mobile, reference, select='bynum 1:10',
niter=10, weights='mass')
assert_almost_equal(
leonwehrhan marked this conversation as resolved.
Show resolved Hide resolved
res.results.positions,
[
[11.96438784, 8.85426235, -10.24735737],
[12.75920431, 8.27294545, -10.54295766],
[12.3285704, 9.72083717, -9.9419435],
[11.33941507, 9.03249423, -11.01306158],
[11.30988499, 8.14958885, -9.1205501],
[12.09108655, 7.85155906, -8.46681943],
[10.37499697, 9.13535837, -8.3732586],
[9.83883314, 8.57939098, -7.6195549],
[9.64405257, 9.55924307, -9.04315991],
[11.0678934, 10.27798773, -7.64881842]
],
decimal=5,
)


def test_alignto_reorder_atomgroups():
# Issue 2977
u = mda.Universe(PDB_helix)
Expand Down
Loading