Source code for galsim.config.bandpass

# Copyright (c) 2012-2023 by the GalSim developers team on GitHub
# https://github.com/GalSim-developers
#
# This file is part of GalSim: The modular galaxy image simulation toolkit.
# https://github.com/GalSim-developers/GalSim
#
# GalSim is free software: redistribution and use in source and binary forms,
# with or without modification, are permitted provided that the following
# conditions are met:
#
# 1. Redistributions of source code must retain the above copyright notice, this
#    list of conditions, and the disclaimer given in the accompanying LICENSE
#    file.
# 2. Redistributions in binary form must reproduce the above copyright notice,
#    this list of conditions, and the disclaimer given in the documentation
#    and/or other materials provided with the distribution.
#

import logging
from astropy.units import Quantity, Unit

from .util import LoggerWrapper
from .value import ParseValue, GetAllParams, GetIndex
from .input import RegisterInputConnectedType
from ..errors import GalSimConfigError, GalSimConfigValueError
from ..bandpass import Bandpass
from ..utilities import basestring

# This module-level dict will store all the registered Bandpass types.
# See the RegisterBandpassType function at the end of this file.
# The keys are the (string) names of the Bandpass types, and the values will be builders that know
# how to build the Bandpass object.
valid_bandpass_types = {}

def BuildBandpass(config, key, base, logger=None):
    """Read the Bandpass parameters from config[key] and return a constructed Bandpass object.

    Parameters:
        config:     A dict with the configuration information.
        key:        The key name in config indicating which object to build.
        base:       The base dict of the configuration.
        logger:     Optionally, provide a logger for logging debug statements. [default: None]

    Returns:
        (bandpass, safe) where bandpass is a Bandpass instance, and safe is whether it is safe to
                         reuse.
    """
    logger = LoggerWrapper(logger)
    logger.debug('obj %d: Start BuildBandpass key = %s',base.get('obj_num',0),key)

    param = config[key]

    # Check for direct value, else get the bandpass type
    if isinstance(param, Bandpass):
        return param, True
    elif isinstance(param, basestring) and (param[0] == '$' or param[0] == '@'):
        return ParseValue(config, key, base, None)
    elif isinstance(param, dict):
        bandpass_type = param.get('type', 'FileBandpass')
    else:
        raise GalSimConfigError("%s must be either a Bandpass or a dict"%key)

    # For these two, just do the usual ParseValue function.
    if bandpass_type in ('Eval', 'Current'):
        return ParseValue(config, key, base, None)

    # Check if we can use the current cached object
    index, index_key = GetIndex(param, base)
    if 'current' in param:
        cbandpass, csafe, cvalue_type, cindex, cindex_key = param['current']
        if cindex == index:
            logger.debug('obj %d: The Bandpass object is already current', base.get('obj_num',0))
            logger.debug('obj %d: index_key = %s, index = %d',base.get('obj_num',0),
                         cindex_key, cindex)
            return cbandpass, csafe

    if bandpass_type not in valid_bandpass_types:
        raise GalSimConfigValueError("Invalid bandpass.type.", bandpass_type,
                                     list(valid_bandpass_types.keys()))
    logger.debug('obj %d: Building bandpass type %s', base.get('obj_num',0), bandpass_type)
    builder = valid_bandpass_types[bandpass_type]
    bandpass, safe = builder.buildBandpass(param, base, logger)
    logger.debug('obj %d: bandpass = %s', base.get('obj_num',0), bandpass)

    param['current'] = bandpass, safe, Bandpass, index, index_key

    return bandpass, safe


[docs]class BandpassBuilder: """A base class for building Bandpass objects. The base class defines the call signatures of the methods that any derived class should follow. """
[docs] def buildBandpass(self, config, base, logger): """Build the Bandpass based on the specifications in the config dict. Note: Sub-classes must override this function with a real implementation. Parameters: config: The configuration dict for the bandpass type. base: The base configuration dict. logger: If provided, a logger for logging debug statements. Returns: the constructed Bandpass object. """ raise NotImplementedError("The %s class has not overridden buildBandpass"%self.__class__)
class FileBandpassBuilder(BandpassBuilder): """A class for loading a Bandpass from a file FileBandpass expects the following parameters: file_name (str) The file to load (required) wave_type (str or Quantity) The units (nm or Ang) of the wavelengths in the file (required) thin (float) A relative error to use for thinning the file (default: None) blue_limit (float or Quantity) A cutoff wavelength on the blue side (default: None) red_limit (float or Quantity) A cutoff wavelength on the red side (default: None) zeropoint (float or str) A zeropoint to use (default: None) """ def buildBandpass(self, config, base, logger): """Build the Bandpass based on the specifications in the config dict. Parameters: config: The configuration dict for the bandpass type. base: The base configuration dict. logger: If provided, a logger for logging debug statements. Returns: the constructed Bandpass object. """ logger = LoggerWrapper(logger) req = { 'file_name': str, 'wave_type': (str, Unit), } opt = { 'thin' : float, 'blue_limit' : (float, Quantity), 'red_limit' : (float, Quantity), 'zeropoint': (float, str) } kwargs, safe = GetAllParams(config, base, req=req, opt=opt) file_name = kwargs.pop('file_name') thin = kwargs.pop('thin', None) zeropoint = kwargs.pop('zeropoint', None) logger.info("Reading Bandpass file: %s",file_name) bandpass = Bandpass(file_name, **kwargs) if zeropoint: bandpass = bandpass.withZeropoint(zeropoint) if thin: bandpass = bandpass.thin(thin) return bandpass, safe
[docs]def RegisterBandpassType(bandpass_type, builder, input_type=None): """Register a bandpass type for use by the config apparatus. Parameters: bandpass_type: The name of the type in the config dict. builder: A builder object to use for building the Bandpass object. It should be an instance of a subclass of BandpassBuilder. input_type: If the Bandpass builder utilises an input object, give the key name of the input type here. (If it uses more than one, this may be a list.) [default: None] """ valid_bandpass_types[bandpass_type] = builder RegisterInputConnectedType(input_type, bandpass_type)
RegisterBandpassType('FileBandpass', FileBandpassBuilder())