Source code for lenstronomy.SimulationAPI.observation_api

import numpy as np
import warnings
import lenstronomy.Util.data_util as data_util
from lenstronomy.Data.psf import PSF

from lenstronomy.Util.package_util import exporter

export, __all__ = exporter()


[docs] @export class Instrument(object): """Basic access points to instrument properties."""
[docs] def __init__(self, pixel_scale, read_noise=None, ccd_gain=None): """ :param read_noise: std of noise generated by read-out (in units of electrons) :param pixel_scale: scale (in arcseconds) of pixels :param ccd_gain: electrons/ADU (analog-to-digital unit). A gain of 8 means that the camera digitizes the CCD signal so that each ADU corresponds to 8 photoelectrons. """ if ccd_gain is not None: ccd_gain = float(ccd_gain) self.ccd_gain = ccd_gain self._read_noise = read_noise self.pixel_scale = pixel_scale
[docs] @export class Observation(object): """Basic access point to observation properties."""
[docs] def __init__( self, exposure_time, sky_brightness=None, seeing=None, num_exposures=1, psf_type="GAUSSIAN", kernel_point_source=None, truncation=5, point_source_supersampling_factor=1, ): """ :param exposure_time: exposure time per image (in seconds) :param sky_brightness: sky brightness (in magnitude per square arcseconds) :param seeing: full width at half maximum of the PSF (if not specific psf_model is specified) :param num_exposures: number of exposures that are combined :param psf_type: string, type of PSF ('GAUSSIAN' and 'PIXEL' supported) :param kernel_point_source: 2d numpy array, model of PSF centered with odd number of pixels per axis (optional when psf_type='PIXEL' is chosen) :param point_source_supersampling_factor: int, supersampling factor of kernel_point_source (optional when psf_type='PIXEL' is chosen) """ self._exposure_time = exposure_time self._sky_brightness_ = sky_brightness self._num_exposures = num_exposures self._seeing = seeing self._psf_type = psf_type self._truncation = truncation self._kernel_point_source = kernel_point_source self._point_source_supersampling_factor = point_source_supersampling_factor
[docs] def update_observation( self, exposure_time=None, sky_brightness=None, seeing=None, num_exposures=None, psf_type=None, kernel_point_source=None, ): """Updates class instance with new properties if specific argument is not None. :param exposure_time: exposure time per image (in seconds) :param sky_brightness: sky brightness (in magnitude per square arcseconds) :param seeing: full width at half maximum of the PSF (if not specific psf_model is specified) :param num_exposures: number of exposures that are combined :param psf_type: string, type of PSF ('GAUSSIAN' and 'PIXEL' supported) :param kernel_point_source: 2d numpy array, model of PSF centered with odd number of pixels per axis (optional when psf_type='PIXEL' is chosen) :return: None, updated class instance """ if exposure_time is not None: self._exposure_time = exposure_time if sky_brightness is not None: self._sky_brightness_ = sky_brightness if seeing is not None: self._seeing = seeing if num_exposures is not None: self._num_exposures = num_exposures if psf_type is not None: self._psf_type = psf_type if kernel_point_source is not None: self._kernel_point_source = kernel_point_source
@property def _sky_brightness(self): if self._sky_brightness_ is None: raise ValueError("sky_brightness is not set in the class instance!") return self._sky_brightness_ @property def exposure_time(self): """Total exposure time. :return: summed exposure time """ return self._exposure_time * self._num_exposures @property def kwargs_psf(self): """Keyword arguments to initiate a PSF() class. :return: kwargs_psf """ if self._psf_type == "GAUSSIAN": psf_type = "GAUSSIAN" fwhm = self._seeing truncation = self._truncation kwargs_psf = {"psf_type": psf_type, "fwhm": fwhm, "truncation": truncation} elif self._psf_type == "PIXEL": if self._kernel_point_source is not None: kwargs_psf = { "psf_type": "PIXEL", "kernel_point_source": self._kernel_point_source, "point_source_supersampling_factor": self._point_source_supersampling_factor, } else: raise ValueError( "You need to create the class instance with a psf_model!" ) elif self._psf_type == "NONE": kwargs_psf = {"psf_type": "NONE"} else: raise ValueError("psf_type %s not supported!" % self._psf_type) return kwargs_psf @property def psf_class(self): """Creates instance of PSF() class based on knowledge of the observations For the full possibility of how to create such an instance, see the PSF() class documentation. :return: instance of PSF() class """ psf_class = PSF(**self.kwargs_psf) return psf_class
[docs] @export class SingleBand(Instrument, Observation): """Class that combines Instrument and Observation."""
[docs] def __init__( self, pixel_scale, exposure_time, magnitude_zero_point, read_noise=None, ccd_gain=None, sky_brightness=None, seeing=None, num_exposures=1, psf_type="GAUSSIAN", kernel_point_source=None, truncation=5, point_source_supersampling_factor=1, data_count_unit="e-", background_noise=None, ): """ :param read_noise: std of noise generated by read-out (in units of electrons) :param pixel_scale: scale (in arcseconds) of pixels :param ccd_gain: electrons/ADU (analog-to-digital unit). A gain of 8 means that the camera digitizes the CCD signal so that each ADU corresponds to 8 photoelectrons. :param exposure_time: exposure time per image (in seconds) :param sky_brightness: sky brightness (in magnitude per square arcseconds in units of electrons) :param seeing: Full-Width-at-Half-Maximum (FWHM) of PSF :param magnitude_zero_point: magnitude corresponding to 1 electron per second :param num_exposures: number of exposures that are combined :param point_source_supersampling_factor: int, supersampling factor of kernel_point_source (optional when psf_type='PIXEL' is chosen) :param data_count_unit: string, unit of the data (not noise properties - see other definitions), 'e-': (electrons assumed to be IID), 'ADU': (analog-to-digital unit) :param background_noise: sqrt(variance of background) as a total contribution from readnoise, sky brightness etc. in units of the data_count_units (e- or ADU) If you set this parameter, it will use this value regardless of the values of read_noise, sky_brightness """ Instrument.__init__( self, pixel_scale, read_noise, ccd_gain ) # read_noise and ccd_gain can be None Observation.__init__( self, exposure_time=exposure_time, sky_brightness=sky_brightness, seeing=seeing, num_exposures=num_exposures, psf_type=psf_type, kernel_point_source=kernel_point_source, point_source_supersampling_factor=point_source_supersampling_factor, truncation=truncation, ) if data_count_unit not in ["e-", "ADU"]: raise ValueError( "count_unit type %s not supported! Please choose e- or ADU." % data_count_unit ) self._data_count_unit = data_count_unit self._background_noise = background_noise self._magnitude_zero_point = magnitude_zero_point
@property def sky_brightness(self): """ :return: sky brightness (counts per square arcseconds in unit of data (e- or ADU's) per unit time) """ cps = self._sky_brightness_cps if self._data_count_unit == "ADU": cps /= self.ccd_gain return cps @property def _sky_brightness_cps(self): """ :return: sky brightness in electrons per second """ cps = data_util.magnitude2cps( self._sky_brightness, magnitude_zero_point=self._magnitude_zero_point ) return cps @property def background_noise(self): """Gaussian sigma of noise level per pixel in counts (e- or ADU) per second. :return: sqrt(variance) of background noise level in data units """ if self._background_noise is None: if self._read_noise is None: raise ValueError( "read_noise is not specified to evaluate background noise!" ) bkg_noise = data_util.bkg_noise( self._read_noise, self._exposure_time, self._sky_brightness_cps, self.pixel_scale, num_exposures=self._num_exposures, ) if self._data_count_unit == "ADU": bkg_noise /= self.ccd_gain return bkg_noise else: if self._read_noise is not None: warnings.warn( "read noise is specified but not used for noise properties. Background noise is estimated" ' from "background_noise" argument.' ) return self._background_noise
[docs] def flux_noise(self, flux): """ :param flux: float or array, units of count_unit/seconds, needs to be positive semi-definite in the flux value :return: Gaussian approximation of Poisson statistics in IIDs sqrt(variance) """ flux_iid = self.flux_iid(flux) variance = ( flux_iid # the variance of a Poisson distribution is the IID count number ) if isinstance(variance, int) or isinstance(variance, float): variance = max(variance, 0) else: variance[ flux_iid < 0 ] = 0 # make sure negative pixels do not lead to variances (or nans) in the return noise = np.sqrt(variance) / self.exposure_time if self._data_count_unit == "ADU": noise /= self.ccd_gain return noise
[docs] def flux_iid(self, flux_per_second): """IID counts. This can be used by lenstronomy to estimate the Poisson errors keeping the assumption that the counts are IIDs (even if they are not). :param flux_per_second: flux count per second in the units set in this class (ADU or e-) :return: IID count number """ if self._data_count_unit == "ADU": exp_time = self.ccd_gain * self.exposure_time else: exp_time = self.exposure_time return exp_time * flux_per_second
[docs] def noise_for_model( self, model, background_noise=True, poisson_noise=True, seed=None ): """ :param model: 2d numpy array of modelled image (with pixels in units of data specified in class) :param background_noise: bool, if True, adds background noise :param poisson_noise: bool, if True, adds Poisson noise of modelled flux :param seed: int, seed number to be used to render the noise properties. If None, then uses the current numpy.random seed to render the noise properties. :return: noise realization corresponding to the model """ if seed is not None: g = np.random.RandomState(seed=seed) else: g = np.random nx, ny = np.shape(model) noise = np.zeros_like(model) if background_noise is True: noise += g.randn(nx, ny) * self.background_noise if poisson_noise is True: noise += g.randn(nx, ny) * self.flux_noise(model) return noise
[docs] def estimate_noise(self, image): """ :param image: noisy data, background subtracted :return: estimated noise map sqrt(variance) for each pixel as estimated from the instrument and observation """ return np.sqrt(self.background_noise**2 + self.flux_noise(image) ** 2)
[docs] def magnitude2cps(self, magnitude): """Converts an apparent magnitude to counts per second (in units of the data) The zero point of an instrument, by definition, is the magnitude of an object that produces one count (or data number, DN) per second. The magnitude of an arbitrary object producing DN counts in an observation of length EXPTIME is therefore: m = -2.5 x log10(DN / EXPTIME) + ZEROPOINT :param magnitude: magnitude of object :return: counts per second of object """ # compute counts in units of e- or ADS (depending on data and magnitude zero point defined) cps = data_util.magnitude2cps( magnitude, magnitude_zero_point=self._magnitude_zero_point ) if self._data_count_unit == "ADU": cps /= self.ccd_gain return cps