File: test_leaflet.py

package info (click to toggle)
mdanalysis 2.10.0-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 116,696 kB
  • sloc: python: 92,135; ansic: 8,156; makefile: 215; sh: 138
file content (201 lines) | stat: -rw-r--r-- 8,479 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
# -*- Mode: python; tab-width: 4; indent-tabs-mode:nil; coding:utf-8 -*-
# vim: tabstop=4 expandtab shiftwidth=4 softtabstop=4 fileencoding=utf-8
#
# 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 Lesser GNU Public Licence, v2.1 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
#
import MDAnalysis
import pytest

from numpy.testing import assert_equal, assert_almost_equal
import numpy as np

from MDAnalysis.analysis.leaflet import LeafletFinder, optimize_cutoff, HAS_NX
from MDAnalysisTests.datafiles import Martini_membrane_gro


LIPID_HEAD_STRING = "name PO4"


@pytest.fixture()
def universe():
    return MDAnalysis.Universe(Martini_membrane_gro)


@pytest.fixture()
def lipid_heads(universe):
    return universe.select_atoms(LIPID_HEAD_STRING)


@pytest.mark.skipif(HAS_NX, reason="networkx is installed")
def test_optional_nx():
    errmsg = "The LeafletFinder class requires an installation of networkx"
    with pytest.raises(ImportError, match=errmsg):
        _ = LeafletFinder(universe, lipid_heads, pbc=True)


@pytest.mark.skipif(not HAS_NX, reason="needs networkx")
class TestLeafletFinder:
    @staticmethod
    def lines2one(lines):
        """Join lines and squash all whitespace"""
        return " ".join(" ".join(lines).split())

    def test_leaflet_finder(self, universe, lipid_heads):
        lfls = LeafletFinder(universe, lipid_heads, pbc=True)
        top_heads, bottom_heads = lfls.groups()
        # Make top be... on top.
        if (
            top_heads.center_of_geometry()[2]
            < bottom_heads.center_of_geometry()[2]
        ):
            top_heads, bottom_heads = (bottom_heads, top_heads)
        assert_equal(
            top_heads.indices,
            np.arange(1, 2150, 12),
            err_msg="Found wrong leaflet lipids",
        )
        assert_equal(
            bottom_heads.indices,
            np.arange(2521, 4670, 12),
            err_msg="Found wrong leaflet lipids",
        )

    def test_string_vs_atomgroup_proper(self, universe, lipid_heads):
        lfls_ag = LeafletFinder(universe, lipid_heads, pbc=True)
        lfls_string = LeafletFinder(universe, LIPID_HEAD_STRING, pbc=True)
        groups_ag = lfls_ag.groups()
        groups_string = lfls_string.groups()
        assert_equal(groups_string[0].indices, groups_ag[0].indices)
        assert_equal(groups_string[1].indices, groups_ag[1].indices)

    def test_optimize_cutoff(self, universe, lipid_heads):
        cutoff, N = optimize_cutoff(universe, lipid_heads, pbc=True)
        assert N == 2
        assert_almost_equal(cutoff, 10.5, decimal=4)

    def test_pbc_on_off(self, universe, lipid_heads):
        lfls_pbc_on = LeafletFinder(universe, lipid_heads, pbc=True)
        lfls_pbc_off = LeafletFinder(universe, lipid_heads, pbc=False)
        assert lfls_pbc_on.graph.size() > lfls_pbc_off.graph.size()

    def test_pbc_on_off_difference(self, universe, lipid_heads):
        import networkx

        lfls_pbc_on = LeafletFinder(universe, lipid_heads, cutoff=7, pbc=True)
        lfls_pbc_off = LeafletFinder(
            universe, lipid_heads, cutoff=7, pbc=False
        )
        pbc_on_graph = lfls_pbc_on.graph
        pbc_off_graph = lfls_pbc_off.graph
        diff_graph = networkx.difference(pbc_on_graph, pbc_off_graph)
        assert_equal(
            set(diff_graph.edges),
            {(69, 153), (73, 79), (206, 317), (313, 319)},
        )

    @pytest.mark.parametrize("sparse", [True, False, None])
    def test_sparse_on_off_none(self, universe, lipid_heads, sparse):
        lfls_ag = LeafletFinder(
            universe, lipid_heads, cutoff=15.0, pbc=True, sparse=sparse
        )
        assert_almost_equal(len(lfls_ag.graph.edges), 1903, decimal=4)

    def test_cutoff_update(self, universe, lipid_heads):
        lfls_ag = LeafletFinder(universe, lipid_heads, cutoff=15.0, pbc=True)
        lfls_ag.update(cutoff=1.0)
        assert_almost_equal(lfls_ag.cutoff, 1.0, decimal=4)
        assert_almost_equal(len(lfls_ag.groups()), 360, decimal=4)

    def test_cutoff_update_default(self, universe, lipid_heads):
        lfls_ag = LeafletFinder(universe, lipid_heads, cutoff=15.0, pbc=True)
        lfls_ag.update()
        assert_almost_equal(lfls_ag.cutoff, 15.0, decimal=4)
        assert_almost_equal(len(lfls_ag.groups()), 2, decimal=4)

    def test_write_selection(self, universe, lipid_heads, tmpdir):
        lfls_ag = LeafletFinder(universe, lipid_heads, cutoff=15.0, pbc=True)
        with tmpdir.as_cwd():
            filename = lfls_ag.write_selection("leaflet.vmd")
            expected_output = self.lines2one(
                [
                    """# leaflets based on select=<AtomGroup with 360 atoms> cutoff=15.000000
            # MDAnalysis VMD selection
            atomselect macro leaflet_1 {index 1 13 25 37 49 61 73 85 \\
            97 109 121 133 145 157 169 181 \\
            193 205 217 229 241 253 265 277 \\
            289 301 313 325 337 349 361 373 \\
            385 397 409 421 433 445 457 469 \\
            481 493 505 517 529 541 553 565 \\
            577 589 601 613 625 637 649 661 \\
            673 685 697 709 721 733 745 757 \\
            769 781 793 805 817 829 841 853 \\
            865 877 889 901 913 925 937 949 \\
            961 973 985 997 1009 1021 1033 1045 \\
            1057 1069 1081 1093 1105 1117 1129 1141 \\
            1153 1165 1177 1189 1201 1213 1225 1237 \\
            1249 1261 1273 1285 1297 1309 1321 1333 \\
            1345 1357 1369 1381 1393 1405 1417 1429 \\
            1441 1453 1465 1477 1489 1501 1513 1525 \\
            1537 1549 1561 1573 1585 1597 1609 1621 \\
            1633 1645 1657 1669 1681 1693 1705 1717 \\
            1729 1741 1753 1765 1777 1789 1801 1813 \\
            1825 1837 1849 1861 1873 1885 1897 1909 \\
            1921 1933 1945 1957 1969 1981 1993 2005 \\
            2017 2029 2041 2053 2065 2077 2089 2101 \\
            2113 2125 2137 2149 }
            # MDAnalysis VMD selection
            atomselect macro leaflet_2 {index 2521 2533 2545 2557 2569 2581 2593 2605 \\
            2617 2629 2641 2653 2665 2677 2689 2701 \\
            2713 2725 2737 2749 2761 2773 2785 2797 \\
            2809 2821 2833 2845 2857 2869 2881 2893 \\
            2905 2917 2929 2941 2953 2965 2977 2989 \\
            3001 3013 3025 3037 3049 3061 3073 3085 \\
            3097 3109 3121 3133 3145 3157 3169 3181 \\
            3193 3205 3217 3229 3241 3253 3265 3277 \\
            3289 3301 3313 3325 3337 3349 3361 3373 \\
            3385 3397 3409 3421 3433 3445 3457 3469 \\
            3481 3493 3505 3517 3529 3541 3553 3565 \\
            3577 3589 3601 3613 3625 3637 3649 3661 \\
            3673 3685 3697 3709 3721 3733 3745 3757 \\
            3769 3781 3793 3805 3817 3829 3841 3853 \\
            3865 3877 3889 3901 3913 3925 3937 3949 \\
            3961 3973 3985 3997 4009 4021 4033 4045 \\
            4057 4069 4081 4093 4105 4117 4129 4141 \\
            4153 4165 4177 4189 4201 4213 4225 4237 \\
            4249 4261 4273 4285 4297 4309 4321 4333 \\
            4345 4357 4369 4381 4393 4405 4417 4429 \\
            4441 4453 4465 4477 4489 4501 4513 4525 \\
            4537 4549 4561 4573 4585 4597 4609 4621 \\
            4633 4645 4657 4669 }
    
    """
                ]
            )

            assert (
                self.lines2one(open("leaflet.vmd").readlines())
                == expected_output
            )

    def test_component_index_is_not_none(self, universe, lipid_heads):
        lfls_ag = LeafletFinder(universe, lipid_heads, cutoff=15.0, pbc=True)
        assert_almost_equal(
            len(lfls_ag.groups(component_index=0)), 180, decimal=4
        )