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