Source code for eofs.tools.cdms

"""Supplementary tools for the `cdms` EOF analysis interface."""
# (c) Copyright 2010-2013 Andrew Dawson. All Rights Reserved.
#
# This file is part of eofs.
#
# eofs is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# eofs is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
# FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
# for more details.
#
# You should have received a copy of the GNU General Public License
# along with eofs.  If not, see <http://www.gnu.org/licenses/>.
from __future__ import (absolute_import, division, print_function)  # noqa

import cdms2
import numpy as np

from .standard import covariance_map as standard_covmap
from .standard import correlation_map as standard_cormap
from .generic import covcor_dimensions


def _rootcoslat_weights(latdim):
    """Square-root of cosine of latitude weights.

    *latdim*
       Latitude dimension values.

    """
    coslat = np.cos(np.deg2rad(latdim))
    coslat[np.where(coslat < 0)] = 0.
    latw = np.sqrt(coslat)
    latw[np.where(np.isnan(latw))] = 0.
    return latw


def _area_weights(grid, gridorder):
    """Area weights.

    *grid*
        `cdms2` grid.

    *gridorder*
        Either "xy" or "yx".

    """
    latw, lonw = grid.getWeights()
    if gridorder == "xy":
        wtarray = np.outer(lonw, latw)
    else:
        wtarray = np.outer(latw, lonw)
    wtarray /= wtarray.sum()
    wtarray = np.sqrt(wtarray)
    return wtarray


[docs]def weights_array(dataset, scheme): """Weights for a data set on a grid. Returned weights are a `numpy.ndarray` broadcastable to the shape of the input data. **Arguments:** *dataset* A `cdms2` variable to generate weights for. *scheme* Weighting scheme to use. The following values are accepted: * *'coslat'* : Square-root of cosine of latitude. * *'area'* : Square-root of grid cell area normalized by total grid area. **Returns:** *weights* An array contanining the weights (not a `cdms2` variable). **Examples:** Area weights for a `cdms2` variable on 2D grid: weights = weights_array(var2d, scheme='area') Square-root of cosine of latitude weights for a `cdms2` variable with a latitude dimension: weights = weights_array(var, scheme='coslat') """ # A re-usable generic error message for the function. When raising an # exception just fill in what is required. errstr = "weighting scheme '{!s}' requires {{!s}}".format(scheme) # Always use lower-case for the scheme, allowing the user to use # upper-case in their calling code without an error. scheme = scheme.lower() if scheme in ('area'): # Handle area weighting. grid = dataset.getGrid() if grid is None: raise ValueError(errstr.format('a grid')) order = dataset.getOrder() if 'xy' in order: gridorder = 'xy' dimtoindex = dataset.getLatitude() elif 'yx' in order: gridorder = 'yx' dimtoindex = dataset.getLongitude() else: raise ValueError( errstr.format('adjacent latitude and longitude dimensions')) # Retrieve area weights for the specified grid. weights = _area_weights(grid, gridorder) elif scheme in ('coslat', 'cos_lat'): # Handle square-root of cosine of latitude weighting. try: latdim = dataset.getLatitude()[:] dimtoindex = dataset.getLatitude() except (AttributeError, TypeError): raise ValueError(errstr.format('a latitude dimension')) # Retrieve latitude weights. weights = _rootcoslat_weights(latdim) else: raise ValueError("invalid weighting scheme: '{!s}'".format(scheme)) # Re-shape the retrieved weights so that they are broadcastable to the # shape of the input arrays. This just involves adding any additional # singleton dimensions to the right of the last weights dimension. rightdims = len(dataset.shape) - dataset.getAxisIndex(dimtoindex) - 1 weights = weights.reshape(weights.shape + (1,) * rightdims) return weights
[docs]def cdms2_name(variable): """Return a human-readable name of a cdms2 variable. First it tries 'standard_name', then 'long_name' then 'id'. """ names = [getattr(variable, 'standard_name', None), getattr(variable, 'long_name', None), getattr(variable, 'id', None)] try: return [name for name in names if name is not None][0] except IndexError: return 'dataset'
[docs]def correlation_map(pcs, field): """Correlation maps for a set of PCs and a spatial-temporal field. Given a set of PCs in a `cdms2` variable (e.g., as output from `eofs.cdms.Eof.pcs`) and a spatial-temporal field in a `cdms` variable, one correlation map per PC is computed. The field must have the same temporal dimension as the PCs. Any number of spatial dimensions (including zero) are allowed in the field and there can be any number of PCs. **Arguments:** *pcs* PCs in a `cdms2` variable. *field* Spatial-temporal field in a `cdms2` variable. **Returns:** *correlation_maps* A `cdms2` variable containing the correlation maps. **Examples:** Compute correlation maps for each PC:: pcs = eofobj.pcs(pcscaling=1) cormaps = correlation_map(pcs, field) """ cor = standard_cormap(pcs.asma(), field.asma()) outdims = covcor_dimensions(pcs.getAxisList(), field.getAxisList()) if not outdims: # There are no output dimensions, return a scalar. return cor # Otherwise return a cdms2 variable with the appropriate dimensions. cor = cdms2.createVariable(cor, axes=outdims, id='pccor') cor.long_name = "pc_correlation" return cor
[docs]def covariance_map(pcs, field, ddof=1): """Covariance maps for a set of PCs and a spatial-temporal field. Given a set of PCs in a `cdms2` variable (e.g., as output from `~eofs.cdms.Eof.pcs`) and a spatial-temporal field in a `cdms2` variable, one covariance map per PC is computed. The field must have the same temporal dimension as the PCs. Any number of spatial dimensions (including zero) are allowed in the field and there can be any number of PCs. **Arguments:** *pcs* PCs in a `cdms2` variable. *field* Spatial-temporal field in a `cdms2` variable. **Optional arguments:** *ddof* 'Delta degrees of freedom'. The divisor used to normalize the covariance matrix is *N - ddof* where *N* is the number of samples. Defaults to *1*. **Returns:** *covariance_maps* A `cdms2` variable containing the covariance maps. **Examples:** Compute covariance maps for each PC:: pcs = eofobj.pcs(pcscaling=1) covmaps = covariance_map(pcs, field) """ cov = standard_covmap(pcs.asma(), field.asma(), ddof=ddof) outdims = covcor_dimensions(pcs.getAxisList(), field.getAxisList()) if not outdims: # There are no output dimensions, return a scalar. return cov # Otherwise return a cdms2 variable with the appropriate dimensions. cov = cdms2.createVariable(cov, axes=outdims, id='pccov') cov.long_name = 'pc_covariance' return cov