Source code for openquake.hazardlib.gsim.multi

# -*- coding: utf-8 -*-
# vim: tabstop=4 shiftwidth=4 softtabstop=4
#
# Copyright (C) 2018-2023 GEM Foundation
#
# OpenQuake is free software: you can redistribute it and/or modify it
# under the terms of the GNU Affero General Public License as published
# by the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# OpenQuake 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 Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with OpenQuake.  If not, see <http://www.gnu.org/licenses/>.
"""
Module exports :class:`MultiGMPE`, which can create a composite of
multiple GMPEs for different IMTs when passed a dictionary of ground motion
models organised by IMT type or by a string describing the association
"""
import numpy as np
from openquake.hazardlib import const, contexts
from openquake.hazardlib.gsim.base import GMPE, registry
from openquake.hazardlib import imt as imt_module

uppernames = '''
DEFINED_FOR_INTENSITY_MEASURE_TYPES
DEFINED_FOR_STANDARD_DEVIATION_TYPES
REQUIRES_SITES_PARAMETERS
REQUIRES_RUPTURE_PARAMETERS
REQUIRES_DISTANCES
'''.split()


[docs]class MultiGMPE(GMPE): """ The MultiGMPE can call ground motions for various IMTs when instantiated with a dictionary of ground motion models organised by IMT or a string describing the association. In the case of spectral accelerations the period of the IMT must be defined explicitly and only SA for that period will be computed. """ #: Supported tectonic region type is undefined DEFINED_FOR_TECTONIC_REGION_TYPE = "" #: Supported intensity measure types are not set DEFINED_FOR_INTENSITY_MEASURE_TYPES = set() #: Supported intensity measure component is horizontal #: :attr:`~openquake.hazardlib.const.IMC.HORIZONTAL`, DEFINED_FOR_INTENSITY_MEASURE_COMPONENT = const.IMC.HORIZONTAL #: Supported standard deviation type DEFINED_FOR_STANDARD_DEVIATION_TYPES = set([const.StdDev.TOTAL]) #: Required site parameters will be set be selected GMPES REQUIRES_SITES_PARAMETERS = set() #: Required rupture parameter is magnitude, others will be set later REQUIRES_RUPTURE_PARAMETERS = {'mag'} #: Required distance metrics will be set by the GMPEs REQUIRES_DISTANCES = set() def __init__(self, **kwargs): """ Instantiate with a dictionary of GMPEs organised by IMT """ super().__init__(**kwargs) for name in uppernames: setattr(self, name, set(getattr(self, name))) for imt, gsim_dic in self.kwargs.items(): [(gsim_name, kw)] = gsim_dic.items() self.kwargs[imt] = gsim = registry[gsim_name](**kw) name = "SA" if imt.startswith("SA") else imt imt_factory = getattr(imt_module, name) if imt_factory not in gsim.DEFINED_FOR_INTENSITY_MEASURE_TYPES: raise ValueError("IMT %s not supported by %s" % (imt, gsim)) for name in uppernames: getattr(self, name).update(getattr(gsim, name)) def __iter__(self): yield from self.kwargs def __getitem__(self, imt): return self.kwargs[imt] def __len__(self): return len(self.kwargs) def __hash__(self): items = tuple((imt, str(gsim)) for imt, gsim in sorted(self.kwargs.items())) return hash(items)
[docs] def compute(self, ctx: np.recarray, imts, mean, sig, tau, phi): """ Call the get mean and stddevs of the GMPE for the respective IMT """ gsims = [self.kwargs[imt.string] for imt in imts] mean[:], sig[:], tau[:], phi[:] = contexts.get_mean_stds( gsims, ctx, imts)