Source code for pyhf.modifiers.histosys

import logging

from . import modifier
from ..paramsets import constrained_by_normal
from .. import get_backend, events
from .. import interpolators

log = logging.getLogger(__name__)


[docs]@modifier(name='histosys', constrained=True, op_code='addition') class histosys(object):
[docs] @classmethod def required_parset(cls, n_parameters): return { 'paramset_type': constrained_by_normal, 'n_parameters': 1, 'modifier': cls.__name__, 'is_constrained': cls.is_constrained, 'is_shared': True, 'inits': (0.0,), 'bounds': ((-5.0, 5.0),), 'auxdata': (0.0,), }
class histosys_combined(object): def __init__(self, histosys_mods, pdfconfig, mega_mods, interpcode='code0'): self._parindices = list(range(len(pdfconfig.suggested_init()))) self.interpcode = interpcode assert self.interpcode in ['code0', 'code2', 'code4p'] pnames = [pname for pname, _ in histosys_mods] keys = ['{}/{}'.format(mtype, m) for m, mtype in histosys_mods] histosys_mods = [m for m, _ in histosys_mods] self._histo_indices = [self._parindices[pdfconfig.par_slice(p)] for p in pnames] self._histosys_histoset = [ [ [ mega_mods[s][m]['data']['lo_data'], mega_mods[s][m]['data']['nom_data'], mega_mods[s][m]['data']['hi_data'], ] for s in pdfconfig.samples ] for m in keys ] self._histosys_mask = [ [[mega_mods[s][m]['data']['mask']] for s in pdfconfig.samples] for m in keys ] if len(histosys_mods): self.interpolator = getattr(interpolators, self.interpcode)( self._histosys_histoset ) self._precompute() events.subscribe('tensorlib_changed')(self._precompute) def _precompute(self): tensorlib, _ = get_backend() self.histosys_mask = tensorlib.astensor(self._histosys_mask) self.histosys_default = tensorlib.zeros(self.histosys_mask.shape) self.histo_indices = tensorlib.astensor(self._histo_indices, dtype='int') def apply(self, pars): tensorlib, _ = get_backend() if not tensorlib.shape(self.histo_indices)[0]: return histosys_alphaset = tensorlib.gather(pars, self.histo_indices) results_histo = self.interpolator(histosys_alphaset) # either rely on numerical no-op or force with line below results_histo = tensorlib.where( self.histosys_mask, results_histo, self.histosys_default ) return results_histo