Source code for MDAnalysis.analysis.hbonds.hbond_analysis

# -*- 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
#

# Hydrogen Bonding Analysis
r"""Hydrogen Bond analysis --- :mod:`MDAnalysis.analysis.hbonds.hbond_analysis`
===========================================================================

:Author: David Caplan, Lukas Grossar, Oliver Beckstein
:Year: 2010-2017
:Copyright: GNU Public License v3


Given a :class:`~MDAnalysis.core.universe.Universe` (simulation
trajectory with 1 or more frames) measure all hydrogen bonds for each
frame between selections 1 and 2.

The :class:`HydrogenBondAnalysis` class is modeled after the `VMD
HBONDS plugin`_.

.. _`VMD HBONDS plugin`: http://www.ks.uiuc.edu/Research/vmd/plugins/hbonds/

Options:
  - *update_selections* (``True``): update selections at each frame?
  - *selection_1_type* ("both"): selection 1 is the: "donor", "acceptor", "both"
  - donor-acceptor *distance* (Å): 3.0
  - Angle *cutoff* (degrees): 120.0
  - *forcefield* to switch between default values for different force fields
  - *donors* and *acceptors* atom types (to add additional atom names)

.. _Analysis Output:

Output
------

The results are
  - the **identities** of donor and acceptor heavy-atoms,
  - the **distance** between the heavy atom acceptor atom and the hydrogen atom
    that is bonded to the heavy atom donor,
  - the **angle** donor-hydrogen-acceptor angle (180º is linear).

Hydrogen bond data are returned per frame, which is stored in
:attr:`HydrogenBondAnalysis.timeseries` (In the following description, ``#``
indicates comments that are not part of the output.)::

    results = [
        [ # frame 1
           [ # hbond 1
              <donor index (0-based)>,
              <acceptor index (0-based)>, <donor string>, <acceptor string>,
              <distance>, <angle>
           ],
           [ # hbond 2
              <donor index (0-based)>,
              <acceptor index (0-based)>, <donor string>, <acceptor string>,
              <distance>, <angle>
           ],
           ....
        ],
        [ # frame 2
          [ ... ], [ ... ], ...
        ],
        ...
    ]

Using the :meth:`HydrogenBondAnalysis.generate_table` method one can reformat
the results as a flat "normalised" table that is easier to import into a
database or dataframe for further processing. The table itself is a
:class:`numpy.recarray`.

.. _Detection-of-hydrogen-bonds:

Detection of hydrogen bonds
---------------------------

Hydrogen bonds are recorded based on a geometric criterion:

1. The distance between acceptor and hydrogen is less than or equal to
   `distance` (default is 3 Å).

2. The angle between donor-hydrogen-acceptor is greater than or equal to
   `angle` (default is 120º).

The cut-off values `angle` and `distance` can be set as keywords to
:class:`HydrogenBondAnalysis`.

Donor and acceptor heavy atoms are detected from atom names. The current
defaults are appropriate for the CHARMM27 and GLYCAM06 force fields as defined
in Table `Default atom names for hydrogen bonding analysis`_.

Hydrogen atoms bonded to a donor are searched with one of two algorithms,
selected with the `detect_hydrogens` keyword.

"distance"
   Searches for all hydrogens (name "H*" or name "[123]H" or type "H") in the
   same residue as the donor atom within a cut-off distance of 1.2 Å.

"heuristic"
   Looks at the next three atoms in the list of atoms following the donor and
   selects any atom whose name matches (name "H*" or name "[123]H"). For

The "distance" search is more rigorous but slower and is set as the
default. Until release 0.7.6, only the heuristic search was implemented.

.. versionchanged:: 0.7.6
   Distance search added (see
   :meth:`HydrogenBondAnalysis._get_bonded_hydrogens_dist`) and heuristic
   search improved (:meth:`HydrogenBondAnalysis._get_bonded_hydrogens_list`)

.. _Default atom names for hydrogen bonding analysis:

.. table:: Default heavy atom names for CHARMM27 force field.

   =========== ==============  =========== ====================================
   group       donor           acceptor    comments
   =========== ==============  =========== ====================================
   main chain  N               O, OC1, OC2 OC1, OC2 from amber99sb-ildn
                                           (Gromacs)
   water       OH2, OW         OH2, OW     SPC, TIP3P, TIP4P (CHARMM27,Gromacs)

   ARG         NE, NH1, NH2
   ASN         ND2             OD1
   ASP                         OD1, OD2
   CYS         SG
   CYH                         SG          possible false positives for CYS
   GLN         NE2             OE1
   GLU                         OE1, OE2
   HIS         ND1, NE2        ND1, NE2    presence of H determines if donor
   HSD         ND1             NE2
   HSE         NE2             ND1
   HSP         ND1, NE2
   LYS         NZ
   MET                         SD          see e.g. [Gregoret1991]_
   SER         OG              OG
   THR         OG1             OG1
   TRP         NE1
   TYR         OH              OH
   =========== ==============  =========== ====================================

.. table:: Heavy atom types for GLYCAM06 force field.

   =========== =========== ==================
   element     donor       acceptor
   =========== =========== ==================
   N           N,NT,N3     N,NT
   O           OH,OW       O,O2,OH,OS,OW,OY
   S                       SM
   =========== =========== ==================

Donor and acceptor names for the CHARMM27 force field will also work for e.g.
OPLS/AA (tested in Gromacs). Residue names in the table are for information
only and are not taken into account when determining acceptors and donors.
This can potentially lead to some ambiguity in the assignment of
donors/acceptors for residues such as histidine or cytosine.

For more information about the naming convention in GLYCAM06 have a look at the
`Carbohydrate Naming Convention in Glycam`_.

.. _`Carbohydrate Naming Convention in Glycam`:
   http://glycam.ccrc.uga.edu/documents/FutureNomenclature.htm

The lists of donor and acceptor names can be extended by providing lists of
atom names in the `donors` and `acceptors` keywords to
:class:`HydrogenBondAnalysis`. If the lists are entirely inappropriate
(e.g. when analysing simulations done with a force field that uses very
different atom names) then one should either use the value "other" for `forcefield`
to set no default values, or derive a new class and set the default list oneself::

 class HydrogenBondAnalysis_OtherFF(HydrogenBondAnalysis):
       DEFAULT_DONORS = {"OtherFF": tuple(set([...]))}
       DEFAULT_ACCEPTORS = {"OtherFF": tuple(set([...]))}

Then simply use the new class instead of the parent class and call it with
`forcefield` = "OtherFF". Please also consider to contribute the list of heavy
atom names to MDAnalysis.

.. rubric:: References

.. [Gregoret1991] L.M. Gregoret, S.D. Rader, R.J. Fletterick, and
   F.E. Cohen. Hydrogen bonds involving sulfur atoms in proteins. Proteins,
   9(2):99–107, 1991. `10.1002/prot.340090204`_.

.. _`10.1002/prot.340090204`: http://dx.doi.org/10.1002/prot.340090204


How to perform HydrogenBondAnalysis
-----------------------------------

All protein-water hydrogen bonds can be analysed with ::

  import MDAnalysis
  import MDAnalysis.analysis.hbonds

  u = MDAnalysis.Universe('topology', 'trajectory')
  h = MDAnalysis.analysis.hbonds.HydrogenBondAnalysis(u, 'protein', 'resname HOH', distance=3.0, angle=120.0)
  h.run()

.. Note::

   Due to the way :class:`HydrogenBondAnalysis` is implemented, it is
   more efficient to have the second selection (`selection2`) be the
   *larger* group, e.g. the water when looking at water-protein
   H-bonds or the whole protein when looking at ligand-protein
   interactions.


The results are stored as the attribute
:attr:`HydrogenBondAnalysis.timeseries`; see :ref:`Analysis Output` for the
format and further options.

A number of convenience functions are provided to process the
:attr:`~HydrogenBondAnalysis.timeseries` according to varying criteria:

:meth:`~HydrogenBondAnalysis.count_by_time`
   time series of the number of hydrogen bonds per time step
:meth:`~HydrogenBondAnalysis.count_by_type`
   data structure with the frequency of each observed hydrogen bond
:meth:`~HydrogenBondAnalysis.timesteps_by_type`
   data structure with a list of time steps for each observed hydrogen bond

For further data analysis it is convenient to process the
:attr:`~HydrogenBondAnalysis.timeseries` data into a normalized table with the
:meth:`~HydrogenBondAnalysis.generate_table` method, which creates a new data
structure :attr:`HydrogenBondAnalysis.table` that contains one row for each
observation of a hydrogen bond::

  h.generate_table()

This table can then be easily turned into, e.g., a `pandas.DataFrame`_, and
further analyzed::

  import pandas as pd
  df = pd.DataFrame.from_records(h.table)

For example, plotting a histogram of the hydrogen bond angles and lengths is as
simple as ::

  df.hist(column=["angle", "distance"])

.. _pandas.DataFrame: http://pandas.pydata.org/pandas-docs/stable/generated/pandas.DataFrame.html


.. TODO: notes on selection updating


Classes
-------

.. autoclass:: HydrogenBondAnalysis
   :members:

   .. attribute:: timesteps

      List of the times of each timestep. This can be used together with
      :attr:`~HydrogenBondAnalysis.timeseries` to find the specific time point
      of a hydrogen bond existence, or see :attr:`~HydrogenBondAnalysis.table`.

   .. attribute:: table

      A normalised table of the data in
      :attr:`HydrogenBondAnalysis.timeseries`, generated by
      :meth:`HydrogenBondAnalysis.generate_table`. It is a
      :class:`numpy.recarray` with the following columns:

          0. "time"
          1. "donor_index"
          2. "acceptor_index"
          3. "donor_resnm"
          4. "donor_resid"
          5. "donor_atom"
          6. "acceptor_resnm"
          7. "acceptor_resid"
          8. "acceptor_atom"
          9. "distance"
          10. "angle"

      It takes up more space than :attr:`~HydrogenBondAnalysis.timeseries` but
      it is easier to analyze and to import into databases or dataframes.


      .. rubric:: Example

      For example, to create a `pandas.DataFrame`_ from ``h.table``::

         import pandas as pd
         df = pd.DataFrame.from_records(h.table)


      .. versionchanged:: 0.17.0
         The 1-based donor and acceptor indices (*donor_idx* and
         *acceptor_idx*) are deprecated in favor of 0-based indices.

   .. automethod:: _get_bonded_hydrogens

   .. automethod:: _get_bonded_hydrogens_dist

   .. automethod:: _get_bonded_hydrogens_list

"""
from __future__ import division, absolute_import
import six
from six.moves import range, zip, map, cPickle

import warnings
import logging
from collections import defaultdict

import numpy as np

from MDAnalysis import MissingDataWarning, NoDataError, SelectionError, SelectionWarning
from .. import base
from MDAnalysis.lib.log import ProgressMeter
from MDAnalysis.lib.NeighborSearch import AtomNeighborSearch
from MDAnalysis.lib import distances
from MDAnalysis.lib.util import deprecate


logger = logging.getLogger('MDAnalysis.analysis.hbonds')


[docs]class HydrogenBondAnalysis(base.AnalysisBase): """Perform a hydrogen bond analysis The analysis of the trajectory is performed with the :meth:`HydrogenBondAnalysis.run` method. The result is stored in :attr:`HydrogenBondAnalysis.timeseries`. See :meth:`~HydrogenBondAnalysis.run` for the format. The default atom names are taken from the CHARMM 27 force field files, which will also work for e.g. OPLS/AA in Gromacs, and GLYCAM06. *Donors* (associated hydrogens are deduced from topology) *CHARMM 27* N of the main chain, water OH2/OW, ARG NE/NH1/NH2, ASN ND2, HIS ND1/NE2, SER OG, TYR OH, CYS SG, THR OG1, GLN NE2, LYS NZ, TRP NE1 *GLYCAM06* N,NT,N3,OH,OW *Acceptors* *CHARMM 27* O of the main chain, water OH2/OW, ASN OD1, ASP OD1/OD2, CYH SG, GLN OE1, GLU OE1/OE2, HIS ND1/NE2, MET SD, SER OG, THR OG1, TYR OH *GLYCAM06* N,NT,O,O2,OH,OS,OW,OY,P,S,SM *amber99sb-ildn(Gromacs)* OC1, OC2 of the main chain See Also -------- :ref:`Default atom names for hydrogen bonding analysis` .. versionchanged:: 0.7.6 DEFAULT_DONORS/ACCEPTORS is now embedded in a dict to switch between default values for different force fields. """ # use tuple(set()) here so that one can just copy&paste names from the # table; set() takes care for removing duplicates. At the end the # DEFAULT_DONORS and DEFAULT_ACCEPTORS should simply be tuples. #: default heavy atom names whose hydrogens are treated as *donors* #: (see :ref:`Default atom names for hydrogen bonding analysis`); #: use the keyword `donors` to add a list of additional donor names. DEFAULT_DONORS = { 'CHARMM27': tuple(set([ 'N', 'OH2', 'OW', 'NE', 'NH1', 'NH2', 'ND2', 'SG', 'NE2', 'ND1', 'NZ', 'OG', 'OG1', 'NE1', 'OH'])), 'GLYCAM06': tuple(set(['N', 'NT', 'N3', 'OH', 'OW'])), 'other': tuple(set([]))} #: default atom names that are treated as hydrogen *acceptors* #: (see :ref:`Default atom names for hydrogen bonding analysis`); #: use the keyword `acceptors` to add a list of additional acceptor names. DEFAULT_ACCEPTORS = { 'CHARMM27': tuple(set([ 'O', 'OC1', 'OC2', 'OH2', 'OW', 'OD1', 'OD2', 'SG', 'OE1', 'OE1', 'OE2', 'ND1', 'NE2', 'SD', 'OG', 'OG1', 'OH'])), 'GLYCAM06': tuple(set(['N', 'NT', 'O', 'O2', 'OH', 'OS', 'OW', 'OY', 'SM'])), 'other': tuple(set([]))} #: A :class:`collections.defaultdict` of covalent radii of common donors #: (used in :meth`_get_bonded_hydrogens_list` to check if a hydrogen is #: sufficiently close to its donor heavy atom). Values are stored for #: N, O, P, and S. Any other heavy atoms are assumed to have hydrogens #: covalently bound at a maximum distance of 1.5 Å. r_cov = defaultdict(lambda: 1.5, # default value N=1.31, O=1.31, P=1.58, S=1.55) def __init__(self, universe, selection1='protein', selection2='all', selection1_type='both', update_selection1=True, update_selection2=True, filter_first=True, distance_type='hydrogen', distance=3.0, angle=120.0, forcefield='CHARMM27', donors=None, acceptors=None, debug=False, detect_hydrogens='distance', verbose=False, pbc=False, **kwargs): """Set up calculation of hydrogen bonds between two selections in a universe. The timeseries is accessible as the attribute :attr:`HydrogenBondAnalysis.timeseries`. Some initial checks are performed. If there are no atoms selected by `selection1` or `selection2` or if no donor hydrogens or acceptor atoms are found then a :exc:`SelectionError` is raised for any selection that does *not* update (`update_selection1` and `update_selection2` keywords). For selections that are set to update, only a warning is logged because it is assumed that the selection might contain atoms at a later frame (e.g. for distance based selections). If no hydrogen bonds are detected or if the initial check fails, look at the log output (enable with :func:`MDAnalysis.start_logging` and set `verbose` ``=True``). It is likely that the default names for donors and acceptors are not suitable (especially for non-standard ligands). In this case, either change the `forcefield` or use customized `donors` and/or `acceptors`. Parameters ---------- universe : Universe Universe object selection1 : str (optional) Selection string for first selection ['protein'] selection2 : str (optional) Selection string for second selection ['all'] selection1_type : {"donor", "acceptor", "both"} (optional) Selection 1 can be 'donor', 'acceptor' or 'both'. Note that the value for `selection1_type` automatically determines how `selection2` handles donors and acceptors: If `selection1` contains 'both' then `selection2` will also contain 'both'. If `selection1` is set to 'donor' then `selection2` is 'acceptor' (and vice versa). ['both']. update_selection1 : bool (optional) Update selection 1 at each frame? Setting to ``False`` is recommended for any static selection to increase performance. [``True``] update_selection2 : bool (optional) Update selection 2 at each frame? Setting to ``False`` is recommended for any static selection to increase performance. [``True``] filter_first : bool (optional) Filter selection 2 first to only atoms 3 * `distance` away [``True``] distance : float (optional) Distance cutoff for hydrogen bonds; only interactions with a H-A distance <= `distance` (and the appropriate D-H-A angle, see `angle`) are recorded. (Note: `distance_type` can change this to the D-A distance.) [3.0] angle : float (optional) Angle cutoff for hydrogen bonds; an ideal H-bond has an angle of 180º. A hydrogen bond is only recorded if the D-H-A angle is >= `angle`. The default of 120º also finds fairly non-specific hydrogen interactions and a possibly better value is 150º. [120.0] forcefield : {"CHARMM27", "GLYCAM06", "other"} (optional) Name of the forcefield used. Switches between different :attr:`~HydrogenBondAnalysis.DEFAULT_DONORS` and :attr:`~HydrogenBondAnalysis.DEFAULT_ACCEPTORS` values. ["CHARMM27"] donors : sequence (optional) Extra H donor atom types (in addition to those in :attr:`~HydrogenBondAnalysis.DEFAULT_DONORS`), must be a sequence. acceptors : sequence (optional) Extra H acceptor atom types (in addition to those in :attr:`~HydrogenBondAnalysis.DEFAULT_ACCEPTORS`), must be a sequence. detect_hydrogens : {"distance", "heuristic"} (optional) Determine the algorithm to find hydrogens connected to donor atoms. Can be "distance" (default; finds all hydrogens in the donor's residue within a cutoff of the donor) or "heuristic" (looks for the next few atoms in the atom list). "distance" should always give the correct answer but "heuristic" is faster, especially when the donor list is updated each for each frame. ["distance"] distance_type : {"hydrogen", "heavy"} (optional) Measure hydrogen bond lengths between donor and acceptor heavy attoms ("heavy") or between donor hydrogen and acceptor heavy atom ("hydrogen"). If using "heavy" then one should set the *distance* cutoff to a higher value such as 3.5 Å. ["hydrogen"] debug : bool (optional) If set to ``True`` enables per-frame debug logging. This is disabled by default because it generates a very large amount of output in the log file. (Note that a logger must have been started to see the output, e.g. using :func:`MDAnalysis.start_logging`.) [``False``] verbose : bool (optional) Toggle progress output. (Can also be given as keyword argument to :meth:`run`.) pbc : bool (optional) Whether to consider periodic boundaries in calculations [``False``] Raises ------ :exc:`SelectionError` is raised for each static selection without the required donors and/or acceptors. Notes ----- In order to speed up processing, atoms are filtered by a coarse distance criterion before a detailed hydrogen bonding analysis is performed (`filter_first` = ``True``). If one of your selections is e.g. the solvent then `update_selection1` (or `update_selection2`) must also be ``True`` so that the list of candidate atoms is updated at each step: this is now the default. If your selections will essentially remain the same for all time steps (i.e. residues are not moving farther than 3 x `distance`), for instance, if neither water nor large conformational changes are involved or if the optimization is disabled (`filter_first` = ``False``) then you can improve performance by setting the `update_selection1` and/or `update_selection2` keywords to ``False``. .. versionchanged:: 0.7.6 New `verbose` keyword (and per-frame debug logging disabled by default). New `detect_hydrogens` keyword to switch between two different algorithms to detect hydrogens bonded to donor. "distance" is a new, rigorous distance search within the residue of the donor atom, "heuristic" is the previous list scan (improved with an additional distance check). New `forcefield` keyword to switch between different values of DEFAULT_DONORS/ACCEPTORS to accomodate different force fields. Also has an option "other" for no default values. .. versionchanged:: 0.8 The new default for `update_selection1` and `update_selection2` is now ``True`` (see `Issue 138`_). Set to ``False`` if your selections only need to be determined once (will increase performance). .. versionchanged:: 0.9.0 New keyword `distance_type` to select between calculation between heavy atoms or hydrogen-acceptor. It defaults to the previous behavior (i.e. "hydrogen"). .. versionchanged:: 0.11.0 Initial checks for selections that potentially raise :exc:`SelectionError`. .. versionchanged:: 0.17.0 use 0-based indexing .. deprecated:: 0.16 The previous `verbose` keyword argument was replaced by `debug`. Note that the `verbose` keyword argument is now consistently used to toggle progress meters throughout the library. .. _`Issue 138`: https://github.com/MDAnalysis/mdanalysis/issues/138 """ super(HydrogenBondAnalysis, self).__init__(universe.trajectory, **kwargs) # per-frame debugging output? self.debug = debug self._get_bonded_hydrogens_algorithms = { "distance": self._get_bonded_hydrogens_dist, # 0.7.6 default "heuristic": self._get_bonded_hydrogens_list, # pre 0.7.6 } if not detect_hydrogens in self._get_bonded_hydrogens_algorithms: raise ValueError("detect_hydrogens must be one of {0!r}".format( self._get_bonded_hydrogens_algorithms.keys())) self.detect_hydrogens = detect_hydrogens self.u = universe self.selection1 = selection1 self.selection2 = selection2 self.selection1_type = selection1_type self.update_selection1 = update_selection1 self.update_selection2 = update_selection2 self.filter_first = filter_first self.distance = distance self.distance_type = distance_type # note: everything except 'heavy' will give the default behavior self.angle = angle self.pbc = pbc and all(self.u.dimensions[:3]) # set up the donors/acceptors lists if donors is None: donors = [] if acceptors is None: acceptors = [] self.forcefield = forcefield self.donors = tuple(set(self.DEFAULT_DONORS[forcefield]).union(donors)) self.acceptors = tuple(set(self.DEFAULT_ACCEPTORS[forcefield]).union(acceptors)) if not (self.selection1 and self.selection2): raise ValueError('HydrogenBondAnalysis: invalid selections') elif self.selection1_type not in ('both', 'donor', 'acceptor'): raise ValueError('HydrogenBondAnalysis: Invalid selection type {0!s}'.format(self.selection1_type)) self._timeseries = None # final result accessed as self.timeseries self.timesteps = None # time for each frame self.table = None # placeholder for output table self._update_selection_1() self._update_selection_2() self._log_parameters() if self.selection1_type == 'donor': self._sanity_check(1, 'donors') self._sanity_check(2, 'acceptors') elif self.selection1_type == 'acceptor': self._sanity_check(1, 'acceptors') self._sanity_check(2, 'donors') else: # both self._sanity_check(1, 'donors') self._sanity_check(1, 'acceptors') self._sanity_check(2, 'acceptors') self._sanity_check(2, 'donors') logger.info("HBond analysis: initial checks passed.") def _sanity_check(self, selection, htype): """sanity check the selections 1 and 2 *selection* is 1 or 2, *htype* is "donors" or "acceptors" If selections do not update and the required donor and acceptor selections are empty then a :exc:`SelectionError` is immediately raised. If selections update dynamically then it is possible that the selection will yield donors/acceptors at a later step and we only issue a warning. .. versionadded:: 0.11.0 """ assert selection in (1, 2) assert htype in ("donors", "acceptors") # horrible data organization: _s1_donors, _s2_acceptors, etc, update_selection1, ... atoms = getattr(self, "_s{0}_{1}".format(selection, htype)) update = getattr(self, "update_selection{0}".format(selection)) if not atoms: errmsg = "No {1} found in selection {0}. " \ "You might have to specify a custom '{1}' keyword.".format( selection, htype) if not update: logger.error(errmsg) raise SelectionError(errmsg) else: errmsg += " Selection will update so continuing with fingers crossed." warnings.warn(errmsg, category=SelectionWarning) logger.warning(errmsg) def _log_parameters(self): """Log important parameters to the logfile.""" logger.info("HBond analysis: selection1 = %r (update: %r)", self.selection1, self.update_selection1) logger.info("HBond analysis: selection2 = %r (update: %r)", self.selection2, self.update_selection2) logger.info("HBond analysis: criterion: donor %s atom and acceptor atom distance <= %.3f A", self.distance_type, self.distance) logger.info("HBond analysis: criterion: angle D-H-A >= %.3f degrees", self.angle) logger.info("HBond analysis: force field %s to guess donor and acceptor names", self.forcefield) logger.info("HBond analysis: bonded hydrogen detection algorithm: %r", self.detect_hydrogens)
[docs] def _get_bonded_hydrogens(self, atom, **kwargs): """Find hydrogens bonded to `atom`. This method is typically not called by a user but it is documented to facilitate understanding of the internals of :class:`HydrogenBondAnalysis`. Parameters ---------- atom : groups.Atom heavy atom **kwargs passed through to the calculation method that was selected with the `detect_hydrogens` kwarg of :class:`HydrogenBondAnalysis`. Returns ------- hydrogen_atoms : AtomGroup or [] list of hydrogens (can be a :class:`~MDAnalysis.core.groups.AtomGroup`) or empty list ``[]`` if none were found. See Also -------- :meth:`_get_bonded_hydrogens_dist` :meth:`_get_bonded_hydrogens_list` .. versionchanged:: 0.7.6 Can switch algorithm by using the `detect_hydrogens` keyword to the constructor. *kwargs* can be used to supply arguments for algorithm. """ return self._get_bonded_hydrogens_algorithms[self.detect_hydrogens](atom, **kwargs)
[docs] def _get_bonded_hydrogens_dist(self, atom): """Find hydrogens bonded within cutoff to `atom`. Hydrogens are detected by either name ("H*", "[123]H*") or type ("H"); this is not fool-proof as the atom type is not always a character but the name pattern should catch most typical occurrences. The distance from `atom` is calculated for all hydrogens in the residue and only those within a cutoff are kept. The cutoff depends on the heavy atom (more precisely, on its element, which is taken as the first letter of its name ``atom.name[0]``) and is parameterized in :attr:`HydrogenBondAnalysis.r_cov`. If no match is found then the default of 1.5 Å is used. Parameters ---------- atom : groups.Atom heavy atom Returns ------- hydrogen_atoms : AtomGroup or [] list of hydrogens (can be a :class:`~MDAnalysis.core.groups.AtomGroup`) or empty list ``[]`` if none were found. Notes ----- The performance of this implementation could be improved once the topology always contains bonded information; it currently uses the selection parser with an "around" selection. .. versionadded:: 0.7.6 """ try: return atom.residue.atoms.select_atoms( "(name H* 1H* 2H* 3H* or type H) and around {0:f} name {1!s}" "".format(self.r_cov[atom.name[0]], atom.name)) except NoDataError: return []
[docs] def _get_bonded_hydrogens_list(self, atom, **kwargs): """Find "bonded" hydrogens to the donor *atom*. At the moment this relies on the **assumption** that the hydrogens are listed directly after the heavy atom in the topology. If this is not the case then this function will fail. Hydrogens are detected by name ``H*``, ``[123]H*`` and they have to be within a maximum distance from the heavy atom. The cutoff distance depends on the heavy atom and is parameterized in :attr:`HydrogenBondAnalysis.r_cov`. Parameters ---------- atom : groups.Atom heavy atom **kwargs ignored Returns ------- hydrogen_atoms : AtomGroup or [] list of hydrogens (can be a :class:`~MDAnalysis.core.groups.AtomGroup`) or empty list ``[]`` if none were found. .. versionchanged:: 0.7.6 Added detection of ``[123]H`` and additional check that a selected hydrogen is bonded to the donor atom (i.e. its distance to the donor is less than the covalent radius stored in :attr:`HydrogenBondAnalysis.r_cov` or the default 1.5 Å). Changed name to :meth:`~HydrogenBondAnalysis._get_bonded_hydrogens_list` and added *kwargs* so that it can be used instead of :meth:`~HydrogenBondAnalysis._get_bonded_hydrogens_dist`. """ warnings.warn("_get_bonded_hydrogens_list() (heuristic detection) does " "not always find " "all hydrogens; Using detect_hydrogens='distance', when " "constructing the HydrogenBondAnalysis class is safer. " "Removal of this feature is targeted for 1.0", category=DeprecationWarning) box = self.u.dimensions if self.pbc else None try: hydrogens = [ a for a in self.u.atoms[atom.index + 1:atom.index + 4] if (a.name.startswith(('H', '1H', '2H', '3H')) and distances.calc_bonds(atom.position, a.position, box=box) < self.r_cov[atom.name[0]]) ] except IndexError: hydrogens = [] # weird corner case that atom is the last one in universe return hydrogens
def _update_selection_1(self): self._s1 = self.u.select_atoms(self.selection1) self.logger_debug("Size of selection 1: {0} atoms".format(len(self._s1))) if not self._s1: logger.warning("Selection 1 '{0}' did not select any atoms." .format(str(self.selection1)[:80])) self._s1_donors = {} self._s1_donors_h = {} self._s1_acceptors = {} if self.selection1_type in ('donor', 'both'): self._s1_donors = self._s1.select_atoms( 'name {0}'.format(' '.join(self.donors))) self._s1_donors_h = {} for i, d in enumerate(self._s1_donors): tmp = self._get_bonded_hydrogens(d) if tmp: self._s1_donors_h[i] = tmp self.logger_debug("Selection 1 donors: {0}".format(len(self._s1_donors))) self.logger_debug("Selection 1 donor hydrogens: {0}".format(len(self._s1_donors_h))) if self.selection1_type in ('acceptor', 'both'): self._s1_acceptors = self._s1.select_atoms( 'name {0}'.format(' '.join(self.acceptors))) self.logger_debug("Selection 1 acceptors: {0}".format(len(self._s1_acceptors))) def _update_selection_2(self): box = self.u.dimensions if self.pbc else None self._s2 = self.u.select_atoms(self.selection2) if self.filter_first and self._s2: self.logger_debug('Size of selection 2 before filtering:' ' {} atoms'.format(len(self._s2))) ns_selection_2 = AtomNeighborSearch(self._s2, box) self._s2 = ns_selection_2.search(self._s1, 3. * self.distance) self.logger_debug('Size of selection 2: {0} atoms'.format(len(self._s2))) if not self._s2: logger.warning('Selection 2 "{0}" did not select any atoms.' .format(str(self.selection2)[:80])) self._s2_donors = {} self._s2_donors_h = {} self._s2_acceptors = {} if not self._s2: return None if self.selection1_type in ('donor', 'both'): self._s2_acceptors = self._s2.select_atoms( 'name {0}'.format(' '.join(self.acceptors))) self.logger_debug("Selection 2 acceptors: {0:d}".format(len(self._s2_acceptors))) if self.selection1_type in ('acceptor', 'both'): self._s2_donors = self._s2.select_atoms( 'name {0}'.format(' '.join(self.donors))) self._s2_donors_h = {} for i, d in enumerate(self._s2_donors): tmp = self._get_bonded_hydrogens(d) if tmp: self._s2_donors_h[i] = tmp self.logger_debug("Selection 2 donors: {0:d}".format(len(self._s2_donors))) self.logger_debug("Selection 2 donor hydrogens: {0:d}".format(len(self._s2_donors_h))) def logger_debug(self, *args): if self.debug: logger.debug(*args)
[docs] def run(self, start=None, stop=None, step=None, verbose=None, **kwargs): """Analyze trajectory and produce timeseries. Stores the hydrogen bond data per frame as :attr:`HydrogenBondAnalysis.timeseries` (see there for output format). Parameters ---------- start : int (optional) starting frame-index for analysis, ``None`` is the first one, 0. `start` and `stop` are 0-based frame indices and are used to slice the trajectory (if supported) [``None``] stop : int (optional) last trajectory frame for analysis, ``None`` is the last one [``None``] step : int (optional) read every `step` between `start` (included) and `stop` (excluded), ``None`` selects 1. [``None``] verbose : bool (optional) toggle progress meter output :class:`~MDAnalysis.lib.log.ProgressMeter` [``True``] debug : bool (optional) enable detailed logging of debugging information; this can create *very big* log files so it is disabled (``False``) by default; setting `debug` toggles the debug status for :class:`HydrogenBondAnalysis`, namely the value of :attr:`HydrogenBondAnalysis.debug`. Other Parameters ---------------- remove_duplicates : bool (optional) duplicate hydrogen bonds are removed from output if set to the default value ``True``; normally, this should not be changed. See Also -------- :meth:`HydrogenBondAnalysis.generate_table` : processing the data into a different format. .. versionchanged:: 0.7.6 Results are not returned, only stored in :attr:`~HydrogenBondAnalysis.timeseries` and duplicate hydrogen bonds are removed from output (can be suppressed with `remove_duplicates` = ``False``) .. versionchanged:: 0.11.0 Accept `quiet` keyword. Analysis will now proceed through frames even if no donors or acceptors were found in a particular frame. .. deprecated:: 0.16 The `quiet` keyword argument is deprecated in favor of the `verbose` one. Previous use of `verbose` now corresponds to the new keyword argument `debug`. """ # sets self.start/stop/step and _pm self._setup_frames(self._trajectory, start, stop, step) logger.info("HBond analysis: starting") logger.debug("HBond analysis: donors %r", self.donors) logger.debug("HBond analysis: acceptors %r", self.acceptors) remove_duplicates = kwargs.pop('remove_duplicates', True) # False: old behaviour if not remove_duplicates: logger.warning("Hidden feature remove_duplicates=False activated: you will probably get duplicate H-bonds.") debug = kwargs.pop('debug', None) if debug is not None and debug != self.debug: self.debug = debug logger.debug("Toggling debug to %r", self.debug) if not self.debug: logger.debug("HBond analysis: For full step-by-step debugging output use debug=True") self._timeseries = [] self.timesteps = [] pm = ProgressMeter(self.n_frames, format="HBonds frame {current_step:5d}: {step:5d}/{numsteps} [{percentage:5.1f}%]\r", verbose=kwargs.get('verbose', False)) try: self.u.trajectory.time def _get_timestep(): return self.u.trajectory.time logger.debug("HBond analysis is recording time step") except NotImplementedError: # chained reader or xyz(?) cannot do time yet def _get_timestep(): return self.u.trajectory.frame logger.warning("HBond analysis is recording frame number instead of time step") logger.info("Starting analysis (frame index start=%d stop=%d, step=%d)", self.start, self.stop, self.step) for progress, ts in enumerate(self.u.trajectory[self.start:self.stop:self.step]): # all bonds for this timestep frame_results = [] # dict of tuples (atom.index, atom.index) for quick check if # we already have the bond (to avoid duplicates) already_found = {} frame = ts.frame timestep = _get_timestep() self.timesteps.append(timestep) pm.echo(progress, current_step=frame) self.logger_debug("Analyzing frame %(frame)d, timestep %(timestep)f ps", vars()) if self.update_selection1: self._update_selection_1() if self.update_selection2: self._update_selection_2() box = self.u.dimensions if self.pbc else None if self.selection1_type in ('donor', 'both') and self._s2_acceptors: self.logger_debug("Selection 1 Donors <-> Acceptors") ns_acceptors = AtomNeighborSearch(self._s2_acceptors, box) for i, donor_h_set in self._s1_donors_h.items(): d = self._s1_donors[i] for h in donor_h_set: res = ns_acceptors.search(h, self.distance) for a in res: angle = distances.calc_angles(d.position, h.position, a.position, box=box) angle = np.rad2deg(angle) donor_atom = h if self.distance_type != 'heavy' else d dist = distances.calc_bonds(donor_atom.position, a.position, box=box) if angle >= self.angle and dist <= self.distance: self.logger_debug( "S1-D: {0!s} <-> S2-A: {1!s} {2:f} A, {3:f} DEG".format(h.index, a.index, dist, angle)) frame_results.append( [h.index, a.index, (h.resname, h.resid, h.name), (a.resname, a.resid, a.name), dist, angle]) already_found[(h.index, a.index)] = True if self.selection1_type in ('acceptor', 'both') and self._s1_acceptors: self.logger_debug("Selection 1 Acceptors <-> Donors") ns_acceptors = AtomNeighborSearch(self._s1_acceptors, box) for i, donor_h_set in self._s2_donors_h.items(): d = self._s2_donors[i] for h in donor_h_set: res = ns_acceptors.search(h, self.distance) for a in res: if remove_duplicates and ( (h.index, a.index) in already_found or (a.index, h.index) in already_found): continue angle = distances.calc_angles(d.position, h.position, a.position, box=box) angle = np.rad2deg(angle) donor_atom = h if self.distance_type != 'heavy' else d dist = distances.calc_bonds(donor_atom.position, a.position, box=box) if angle >= self.angle and dist <= self.distance: self.logger_debug( "S1-A: {0!s} <-> S2-D: {1!s} {2:f} A, {3:f} DEG".format(a.index, h.index, dist, angle)) frame_results.append( [h.index, a.index, (h.resname, h.resid, h.name), (a.resname, a.resid, a.name), dist, angle]) self._timeseries.append(frame_results) logger.info("HBond analysis: complete; timeseries %s.timeseries", self.__class__.__name__)
@property def timeseries(self): """Time series of hydrogen bonds. The results of the hydrogen bond analysis can be accessed as a `list` of `list` of `list`: 1. `timeseries[i]`: data for the i-th trajectory frame (at time `timesteps[i]`, see :attr:`timesteps`) 2. `timeseries[i][j]`: j-th hydrogen bond that was detected at the i-th frame. 3. ``donor_index, acceptor_index, donor_name_str, acceptor_name_str, distance, angle = timeseries[i][j]``: structure of one hydrogen bond data item In the following description, ``#`` indicates comments that are not part of the output:: results = [ [ # frame 1 [ # hbond 1 <donor index (0-based)>, <acceptor index (0-based)>, <donor string>, <acceptor string>, <distance>, <angle> ], [ # hbond 2 <donor index (0-based)>, <acceptor index (0-based)>, <donor string>, <acceptor string>, <distance>, <angle> ], .... ], [ # frame 2 [ ... ], [ ... ], ... ], ... ] The time of each step is not stored with each hydrogen bond frame but in :attr:`~HydrogenBondAnalysis.timesteps`. Note ---- For instance, to find an acceptor atom in :attr:`Universe.atoms` by *index* one would use ``u.atoms[acceptor_index]``. The :attr:`timeseries` is a managed attribute and it is generated from the underlying data in :attr:`_timeseries` every time the attribute is accessed. It is therefore costly to call and if :attr:`timeseries` is needed repeatedly it is recommended that you assign to a variable:: h = HydrogenBondAnalysis(u) h.run() timeseries = h.timeseries See Also -------- :attr:`table` : structured array of the data .. versionchanged:: 0.16.1 :attr:`timeseries` has become a managed attribute and is generated from the stored :attr:`_timeseries` when needed. :attr:`_timeseries` contains the donor atom and acceptor atom specifiers as tuples `(resname, resid, atomid)` instead of strings. .. versionchanged:: 0.17.0 The 1-based indices "donor_idx" and "acceptor_idx" are being removed in favor of the 0-based indices "donor_index" and "acceptor_index". """ return [[self._reformat_hb(hb) for hb in hframe] for hframe in self._timeseries] @staticmethod def _reformat_hb(hb, atomformat="{0[0]!s}{0[1]!s}:{0[2]!s}"): """Convert 0.16.1 _timeseries hbond item to 0.16.0 hbond item. In 0.16.1, donor and acceptor are stored as a tuple(resname, resid, atomid). In 0.16.0 and earlier they were stored as a string. .. deprecated:: 1.0 This is a compatibility layer so that we can provide the same output in timeseries as before. However, for 1.0 we should make timeseries just return _timeseries, i.e., change the format of timeseries to the un-ambiguous representation provided in _timeseries. """ return (hb[:2] + [atomformat.format(hb[2]), atomformat.format(hb[3])] + hb[4:])
[docs] def generate_table(self): """Generate a normalised table of the results. The table is stored as a :class:`numpy.recarray` in the attribute :attr:`~HydrogenBondAnalysis.table`. See Also -------- HydrogenBondAnalysis.table """ if self._timeseries is None: msg = "No timeseries computed, do run() first." warnings.warn(msg, category=MissingDataWarning) logger.warning(msg) return num_records = np.sum([len(hframe) for hframe in self._timeseries]) # build empty output table dtype = [ ("time", float), ("donor_index", int), ("acceptor_index", int), ("donor_resnm", "|U4"), ("donor_resid", int), ("donor_atom", "|U4"), ("acceptor_resnm", "|U4"), ("acceptor_resid", int), ("acceptor_atom", "|U4"), ("distance", float), ("angle", float)] # according to Lukas' notes below, using a recarray at this stage is ineffective # and speedups of ~x10 can be achieved by filling a standard array, like this: out = np.empty((num_records,), dtype=dtype) cursor = 0 # current row for t, hframe in zip(self.timesteps, self._timeseries): for (donor_index, acceptor_index, donor, acceptor, distance, angle) in hframe: # donor|acceptor = (resname, resid, atomid) out[cursor] = (t, donor_index, acceptor_index) + \ donor + acceptor + (distance, angle) cursor += 1 assert cursor == num_records, "Internal Error: Not all HB records stored" self.table = out.view(np.recarray) logger.debug("HBond: Stored results as table with %(num_records)d entries.", vars())
@deprecate(release="0.19.0", remove="1.0.0", message="You can instead use ``np.save(filename, " "HydrogendBondAnalysis.table)``.") def save_table(self, filename="hbond_table.pickle"): """Saves :attr:`~HydrogenBondAnalysis.table` to a pickled file. If :attr:`~HydrogenBondAnalysis.table` does not exist yet, :meth:`generate_table` is called first. Parameters ---------- filename : str (optional) path to the filename Example ------- Load with :: import cPickle table = cPickle.load(open(filename)) """ if self.table is None: self.generate_table() with open(filename, 'w') as f: cPickle.dump(self.table, f, protocol=cPickle.HIGHEST_PROTOCOL) def _has_timeseries(self): has_timeseries = self._timeseries is not None if not has_timeseries: msg = "No timeseries computed, do run() first." warnings.warn(msg, category=MissingDataWarning) logger.warning(msg) return has_timeseries
[docs] def count_by_time(self): """Counts the number of hydrogen bonds per timestep. Processes :attr:`HydrogenBondAnalysis._timeseries` into the time series ``N(t)`` where ``N`` is the total number of observed hydrogen bonds at time ``t``. Returns ------- counts : numpy.recarray The resulting array can be thought of as rows ``(time, N)`` where ``time`` is the time (in ps) of the time step and ``N`` is the total number of hydrogen bonds. """ if not self._has_timeseries(): return out = np.empty((len(self.timesteps),), dtype=[('time', float), ('count', int)]) for cursor, time_count in enumerate(zip(self.timesteps, (len(series) for series in self._timeseries))): out[cursor] = time_count return out.view(np.recarray)
[docs] def count_by_type(self): """Counts the frequency of hydrogen bonds of a specific type. Processes :attr:`HydrogenBondAnalysis._timeseries` and returns a :class:`numpy.recarray` containing atom indices, residue names, residue numbers (for donors and acceptors) and the fraction of the total time during which the hydrogen bond was detected. Returns ------- counts : numpy.recarray Each row of the array contains data to define a unique hydrogen bond together with the frequency (fraction of the total time) that it has been observed. .. versionchanged:: 0.17.0 The 1-based indices "donor_idx" and "acceptor_idx" are being deprecated in favor of zero-based indices. """ if not self._has_timeseries(): return hbonds = defaultdict(int) for hframe in self._timeseries: for (donor_index, acceptor_index, donor, acceptor, distance, angle) in hframe: donor_resnm, donor_resid, donor_atom = donor acceptor_resnm, acceptor_resid, acceptor_atom = acceptor # generate unambigous key for current hbond \ # (the donor_heavy_atom placeholder '?' is added later) # idx_zero is redundant for an unambigous key, but included for # consistency. hb_key = ( donor_index, acceptor_index, donor_resnm, donor_resid, "?", donor_atom, acceptor_resnm, acceptor_resid, acceptor_atom) hbonds[hb_key] += 1 # build empty output table dtype = [ ("donor_index", int), ("acceptor_index", int), ('donor_resnm', 'U4'), ('donor_resid', int), ('donor_heavy_atom', 'U4'), ('donor_atom', 'U4'), ('acceptor_resnm', 'U4'), ('acceptor_resid', int), ('acceptor_atom', 'U4'), ('frequency', float) ] out = np.empty((len(hbonds),), dtype=dtype) # float because of division later tsteps = float(len(self.timesteps)) for cursor, (key, count) in enumerate(six.iteritems(hbonds)): out[cursor] = key + (count / tsteps,) # return array as recarray # The recarray has not been used within the function, because accessing the # the elements of a recarray (3.65 us) is much slower then accessing those # of a ndarray (287 ns). r = out.view(np.recarray) # patch in donor heavy atom names (replaces '?' in the key) h2donor = self._donor_lookup_table_byindex() r.donor_heavy_atom[:] = [h2donor[idx] for idx in r.donor_index] return r
[docs] def timesteps_by_type(self): """Frames during which each hydrogen bond existed, sorted by hydrogen bond. Processes :attr:`HydrogenBondAnalysis._timeseries` and returns a :class:`numpy.recarray` containing atom indices, residue names, residue numbers (for donors and acceptors) and each timestep at which the hydrogen bond was detected. In principle, this is the same as :attr:`~HydrogenBondAnalysis.table` but sorted by hydrogen bond and with additional data for the *donor_heavy_atom* and angle and distance omitted. Returns ------- data : numpy.recarray .. versionchanged:: 0.17.0 The 1-based indices "donor_idx" and "acceptor_idx" are being replaced in favor of zero-based indices. """ if not self._has_timeseries(): return hbonds = defaultdict(list) for (t, hframe) in zip(self.timesteps, self._timeseries): for (donor_index, acceptor_index, donor, acceptor, distance, angle) in hframe: donor_resnm, donor_resid, donor_atom = donor acceptor_resnm, acceptor_resid, acceptor_atom = acceptor # generate unambigous key for current hbond # (the donor_heavy_atom placeholder '?' is added later) # idx_zero is redundant for key but added for consistency hb_key = ( donor_index, acceptor_index, donor_resnm, donor_resid, "?", donor_atom, acceptor_resnm, acceptor_resid, acceptor_atom) hbonds[hb_key].append(t) out_nrows = 0 # count number of timesteps per key to get length of output table for ts_list in six.itervalues(hbonds): out_nrows += len(ts_list) # build empty output table dtype = [ ('donor_index', int), ('acceptor_index', int), ('donor_resnm', 'U4'), ('donor_resid', int), ('donor_heavy_atom', 'U4'), ('donor_atom', 'U4'),('acceptor_resnm', 'U4'), ('acceptor_resid', int), ('acceptor_atom', 'U4'), ('time', float)] out = np.empty((out_nrows,), dtype=dtype) out_row = 0 for (key, times) in six.iteritems(hbonds): for tstep in times: out[out_row] = key + (tstep,) out_row += 1 # return array as recarray # The recarray has not been used within the function, because accessing the # the elements of a recarray (3.65 us) is much slower then accessing those # of a ndarray (287 ns). r = out.view(np.recarray) # patch in donor heavy atom names (replaces '?' in the key) h2donor = self._donor_lookup_table_byindex() r.donor_heavy_atom[:] = [h2donor[idx] for idx in r.donor_index] return r
def _donor_lookup_table_byres(self): """Look-up table to identify the donor heavy atom from resid and hydrogen name. Assumptions: * resids are unique * hydrogen atom names are unique within a residue * selections have not changed (because we are simply looking at the last content of the donors and donor hydrogen lists) Donors from `selection1` and `selection2` are merged. Output dictionary ``h2donor`` can be used as:: heavy_atom_name = h2donor[resid][hydrogen_name] """ s1d = self._s1_donors # list of donor Atom instances s1h = self._s1_donors_h # dict indexed by donor position in donor list, containg AtomGroups of H s2d = self._s2_donors s2h = self._s2_donors_h def _make_dict(donors, hydrogens): # two steps so that entry for one residue can be UPDATED for multiple donors d = dict((donors[k].resid, {}) for k in range(len(donors)) if k in hydrogens) for k in range(len(donors)): if k in hydrogens: d[donors[k].resid].update(dict((atom.name, donors[k].name) for atom in hydrogens[k])) return d h2donor = _make_dict(s2d, s2h) # 2 is typically the larger group # merge (in principle h2donor.update(_make_dict(s1d, s1h) should be sufficient # with our assumptions but the following should be really safe) for resid, names in _make_dict(s1d, s1h).items(): if resid in h2donor: h2donor[resid].update(names) else: h2donor[resid] = names return h2donor def _donor_lookup_table_byindex(self): """Look-up table to identify the donor heavy atom from hydrogen atom index. Assumptions: * selections have not changed (because we are simply looking at the last content of the donors and donor hydrogen lists) Donors from `selection1` and `selection2` are merged. Output dictionary ``h2donor`` can be used as:: heavy_atom_name = h2donor[index] """ s1d = self._s1_donors # list of donor Atom instances s1h = self._s1_donors_h # dict indexed by donor position in donor list, containg AtomGroups of H s2d = self._s2_donors s2h = self._s2_donors_h def _make_dict(donors, hydrogens): #return dict(flatten_1([(atom.id, donors[k].name) for atom in hydrogens[k]] for k in range(len(donors)) # if k in hydrogens)) x = [] for k in range(len(donors)): if k in hydrogens: x.extend([(atom.index, donors[k].name) for atom in hydrogens[k]]) return dict(x) h2donor = _make_dict(s2d, s2h) # 2 is typically the larger group h2donor.update(_make_dict(s1d, s1h)) return h2donor