Source code for sherpa.astro.background

#
#  Copyright (C) 2007, 2020, 2021  Smithsonian Astrophysical Observatory
#
#
#  This program 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.
#
#  This program 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 this program; if not, write to the Free Software Foundation, Inc.,
#  51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
#

"""Support multiple background components for a PHA dataset.

Functions in this module might also be useful without the presence of a
background.
"""

from collections import defaultdict
import logging

import numpy as np

from sherpa.astro.instrument import PileupResponse1D
from sherpa.models.model import ArithmeticConstantModel
from sherpa.utils.err import ModelErr


__all__ = ('add_response', 'get_response_for_pha')


warning = logging.getLogger(__name__).warning


[docs] def add_response(session, id, data, model): """Create the response model describing the source and model. Include any background components and apply the response model for the dataset. Parameters ---------- session : sherpa.astro.ui.utils.Session instance id : int or str The identifier for the dataset. data : sherpa.astro.data.DataPHA instance The dataset (may be a background dataset). model : sherpa.models.model.ArithmeticModel instance The model (without response or background components) to match to data. Returns ------- fullmodel : sherpa.models.model.ArithmeticModel The model including the necessary response models and background components. """ id = session._fix_id(id) # QUS: if this gets used to generate the response for the # background then how does it pick up the correct response # (ie when fit_bkg is used). Or does that get generated # by a different code path? pileup_model = session._pileup_models.get(id) bkg_srcs = session._background_sources.get(id, {}) return get_response_for_pha(data, model, bkg_srcs, pileup_model, id)
[docs] def get_response_for_pha(data, model, bkg_srcs={}, pileup_model=None, id=None): """Create the response model describing the source and model. Include any background components and apply the response model for the dataset. This is essentially the object-oriented version of `sherpa.astro.background.add_response`. Parameters ---------- data : sherpa.astro.data.DataPHA instance The dataset (may be a background dataset). model : sherpa.models.model.ArithmeticModel instance The model (without response or background components) to match to data. bkg_srcs : dict Keys in the dictionary need to be the background ids in the dataset ``data``, and the values are the corresponding source models. pileup_model : None or `sherpa.astro.models.JDPileup` instance Pileup model for the dataset if needed, or ``None`` for no pileup model. id : string A string to label the dataset in warning messages. If this is set ``None`` the name of the dataset it used. Thus parameters is mainly needed if this function is called from the UI layer, where datasets and models have ids that are not stored in an attribute of the dataset itself. Returns ------- fullmodel : sherpa.models.model.ArithmeticModel The model including the necessary response models and background components. """ if id is None: id = data.name resp = data.get_full_response(pileup_model) if data.subtracted or (len(bkg_srcs) == 0): return resp(model) # At this point we have background one or more background # components that need to be added to the overall model. # If the scale factors are all scalars then we can return # # resp(model + sum (scale_i * bgnd_i)) [1] # # but if any are arrays then we have to apply the scale factor # after applying the response, that is # # resp(model) + sum(scale_i * resp(bgnd_i)) [2] # # This is because the scale values are in channel space, # and not the instrument response (i.e. the values used inside # the resp call). # # Note that if resp is not a linear response - for instance, # it is a pileup model - then we will not get the correct # answer if there's an array value for the scale factor # (i.e. equation [2] above). A warning message is created in this # case, but it is not treated as an error. # # For multiple background datasets we can have different models - # that is, one for each dataset - but it is expected that the # same model is used for all backgrounds (i.e. this is what # we 'optimise' for). # Identify the scalar and vector scale values for each # background dataset, and combine using the model as a key. # scales_scalar = defaultdict(list) scales_vector = defaultdict(list) for bkg_id in data.background_ids: try: bmdl = bkg_srcs[bkg_id] except KeyError: raise ModelErr('nobkg', bkg_id, id) scale = data.get_background_scale(bkg_id, units='rate', group=False) if np.isscalar(scale): store = scales_scalar else: store = scales_vector store[bmdl].append(scale) # Combine the scalar terms, grouping by the model. # for mdl, scales in scales_scalar.items(): scale = sum(scales) model += scale * mdl # Apply the instrument response. # model = resp(model) if len(scales_vector) == 0: return model # Warn if a pileup model is being used. The error message here # is terrible. # # Should this be a Python Warning rather than a logged message? # if isinstance(resp, PileupResponse1D): wmsg = "model results for dataset {} ".format(id) + \ "likely wrong: use of pileup model and array scaling " + \ "for the background" # warnings.warn(wmsg) warning(wmsg) # Combine the vector terms, grouping by the model. A trick here # is that,to make the string version of the model be readable, # we add a model to contain the scale values, using the # ArithmeticConstantModel. # # Note that the model is given a name, to make it "easy" to # read in the model expression, but this name is not registered # anywhere. An alternative would be to use the default naming # convention of the model, which will use 'float64[n]' as a label. # nvectors = len(scales_vector) for i, (mdl, scales) in enumerate(scales_vector.items(), 1): # special case the single-value case if nvectors == 1: name = 'scale{}'.format(id) else: name = 'scale{}_{}'.format(id, i) # We sum up the scale arrays for this model. # scale = sum(scales) tbl = ArithmeticConstantModel(scale, name=name) model += tbl * resp(mdl) return model