def test_raise(self): with self.assertRaises(ValueError): MultiPlaneBase(z_source_convention=1, lens_model_list=['SIS'], lens_redshift_list=[2]) with self.assertRaises(ValueError): MultiPlaneBase(z_source_convention=1, lens_model_list=['SIS', 'SIS'], lens_redshift_list=[0.5]) with self.assertRaises(ValueError): lens = MultiPlane(z_source_convention=1, z_source=1, lens_model_list=['SIS', 'SIS'], lens_redshift_list=[0.5, 0.8]) lens._check_raise(k=[1]) with self.assertRaises(ValueError): lens_model = LensModel(lens_model_list=['SIS'], multi_plane=True, z_source=1, z_source_convention=1, cosmo_interp=True, z_interp_stop=0.5)
def __init__(self, z_source, lens_model_list, lens_redshift_list, cosmo=None, numerical_alpha_class=None, observed_convention_index=None, ignore_observed_positions=False, z_source_convention=None, cosmo_interp=False, z_interp_stop=None, num_z_interp=100, kwargs_interp=None): """ :param z_source: source redshift for default computation of reduced lensing quantities :param lens_model_list: list of lens model strings :param lens_redshift_list: list of floats with redshifts of the lens models indicated in lens_model_list :param cosmo: instance of astropy.cosmology :param numerical_alpha_class: an instance of a custom class for use in NumericalAlpha() lens model (see documentation in Profiles/numerical_alpha) :param kwargs_interp: interpolation keyword arguments specifying the numerics. See description in the Interpolate() class. Only applicable for 'INTERPOL' and 'INTERPOL_SCALED' models. :param observed_convention_index: a list of indices, corresponding to the lens_model_list element with same index, where the 'center_x' and 'center_y' kwargs correspond to observed (lensed) positions, not physical positions. The code will compute the physical locations when performing computations :param ignore_observed_positions: bool, if True, will ignore the conversion between observed to physical position of deflectors :param z_source_convention: float, redshift of a source to define the reduced deflection angles of the lens models. If None, 'z_source' is used. """ if z_source_convention is None: z_source_convention = z_source if z_interp_stop is None: z_interp_stop = max(z_source, z_source_convention) if z_interp_stop < max(z_source, z_source_convention): raise ValueError( 'z_interp_stop= %s needs to be larger or equal the maximum of z_source=%s and z_source_convention=%s' % (z_interp_stop, z_source, z_source_convention)) self._multi_plane_base = MultiPlaneBase( lens_model_list=lens_model_list, lens_redshift_list=lens_redshift_list, cosmo=cosmo, numerical_alpha_class=numerical_alpha_class, z_source_convention=z_source_convention, cosmo_interp=cosmo_interp, z_interp_stop=z_interp_stop, num_z_interp=num_z_interp, kwargs_interp=kwargs_interp) self._set_source_distances(z_source) self._observed_convention_index = observed_convention_index if observed_convention_index is None: self._convention = PhysicalLocation() else: assert isinstance(observed_convention_index, list) self._convention = LensedLocation(self._multi_plane_base, observed_convention_index) self.ignore_observed_positions = ignore_observed_positions
class MultiPlane(object): """ Multi-plane lensing class with option to assign positions of a selected set of lens models in the observed plane. The lens model deflection angles are in units of reduced deflections from the specified redshift of the lens to the source redshift of the class instance. """ def __init__(self, z_source, lens_model_list, lens_redshift_list, cosmo=None, numerical_alpha_class=None, observed_convention_index=None, ignore_observed_positions=False, z_source_convention=None, cosmo_interp=False, z_interp_stop=None, num_z_interp=100, kwargs_interp=None): """ :param z_source: source redshift for default computation of reduced lensing quantities :param lens_model_list: list of lens model strings :param lens_redshift_list: list of floats with redshifts of the lens models indicated in lens_model_list :param cosmo: instance of astropy.cosmology :param numerical_alpha_class: an instance of a custom class for use in NumericalAlpha() lens model (see documentation in Profiles/numerical_alpha) :param kwargs_interp: interpolation keyword arguments specifying the numerics. See description in the Interpolate() class. Only applicable for 'INTERPOL' and 'INTERPOL_SCALED' models. :param observed_convention_index: a list of indices, corresponding to the lens_model_list element with same index, where the 'center_x' and 'center_y' kwargs correspond to observed (lensed) positions, not physical positions. The code will compute the physical locations when performing computations :param ignore_observed_positions: bool, if True, will ignore the conversion between observed to physical position of deflectors :param z_source_convention: float, redshift of a source to define the reduced deflection angles of the lens models. If None, 'z_source' is used. """ if z_source_convention is None: z_source_convention = z_source if z_interp_stop is None: z_interp_stop = max(z_source, z_source_convention) if z_interp_stop < max(z_source, z_source_convention): raise ValueError( 'z_interp_stop= %s needs to be larger or equal the maximum of z_source=%s and ' 'z_source_convention=%s' % (z_interp_stop, z_source, z_source_convention)) self._multi_plane_base = MultiPlaneBase( lens_model_list=lens_model_list, lens_redshift_list=lens_redshift_list, cosmo=cosmo, numerical_alpha_class=numerical_alpha_class, z_source_convention=z_source_convention, cosmo_interp=cosmo_interp, z_interp_stop=z_interp_stop, num_z_interp=num_z_interp, kwargs_interp=kwargs_interp) self._set_source_distances(z_source) self._observed_convention_index = observed_convention_index if observed_convention_index is None: self._convention = PhysicalLocation() else: assert isinstance(observed_convention_index, list) self._convention = LensedLocation(self._multi_plane_base, observed_convention_index) self.ignore_observed_positions = ignore_observed_positions def update_source_redshift(self, z_source): """ update instance of this class to compute reduced lensing quantities and time delays to a specific source redshift :param z_source: float; source redshift :return: self variables update to new redshift """ if z_source == self._z_source: pass else: self._set_source_distances(z_source) def _set_source_distances(self, z_source): """ compute the relevant angular diameter distances to a specific source redshift :param z_source: float, source redshift :return: self variables """ self._z_source = z_source self._T_ij_start, self._T_ij_stop = self._multi_plane_base.transverse_distance_start_stop( z_start=0, z_stop=z_source, include_z_start=False) self._T_z_source = self._multi_plane_base._cosmo_bkg.T_xy(0, z_source) def observed2flat_convention(self, kwargs_lens): """ :param kwargs_lens: keyword argument list of lens model parameters in the observed convention :return: kwargs_lens positions mapped into angular position without lensing along its LOS """ return self._convention(kwargs_lens) def ray_shooting(self, theta_x, theta_y, kwargs_lens, check_convention=True, k=None): """ ray-tracing (backwards light cone) to the default z_source redshift :param theta_x: angle in x-direction on the image (usually arc seconds, in the same convention as lensing deflection angles) :param theta_y: angle in y-direction on the image (usually arc seconds, in the same convention as lensing deflection angles) :param kwargs_lens: lens model keyword argument list :param check_convention: flag to check the image position convention (leave this alone) :return: angles in the source plane """ self._check_raise(k=k) if check_convention and not self.ignore_observed_positions: kwargs_lens = self._convention(kwargs_lens) x = np.zeros_like(theta_x, dtype=float) y = np.zeros_like(theta_y, dtype=float) alpha_x = np.array(theta_x) alpha_y = np.array(theta_y) x, y, _, _ = self._multi_plane_base.ray_shooting_partial( x, y, alpha_x, alpha_y, z_start=0, z_stop=self._z_source, kwargs_lens=kwargs_lens, T_ij_start=self._T_ij_start, T_ij_end=self._T_ij_stop) beta_x, beta_y = self.co_moving2angle_source(x, y) return beta_x, beta_y def ray_shooting_partial(self, x, y, alpha_x, alpha_y, z_start, z_stop, kwargs_lens, include_z_start=False, check_convention=True, T_ij_start=None, T_ij_end=None): """ ray-tracing through parts of the coin, starting with (x,y) co-moving distances and angles (alpha_x, alpha_y) at redshift z_start and then backwards to redshift z_stop :param x: co-moving position [Mpc] / angle definition :param y: co-moving position [Mpc] / angle definition :param alpha_x: ray angle at z_start [arcsec] :param alpha_y: ray angle at z_start [arcsec] :param z_start: redshift of start of computation :param z_stop: redshift where output is computed :param kwargs_lens: lens model keyword argument list :param include_z_start: bool, if True, includes the computation of the deflection angle at the same redshift as the start of the ray-tracing. ATTENTION: deflection angles at the same redshift as z_stop will be computed! This can lead to duplications in the computation of deflection angles. :param check_convention: flag to check the image position convention (leave this alone) :param T_ij_start: transverse angular distance between the starting redshift to the first lens plane to follow. If not set, will compute the distance each time this function gets executed. :param T_ij_end: transverse angular distance between the last lens plane being computed and z_end. If not set, will compute the distance each time this function gets executed. :return: co-moving position (modulo angle definition) and angles at redshift z_stop """ if check_convention and not self.ignore_observed_positions: kwargs_lens = self._convention(kwargs_lens) return self._multi_plane_base.ray_shooting_partial( x, y, alpha_x, alpha_y, z_start, z_stop, kwargs_lens, include_z_start=include_z_start, T_ij_start=T_ij_start, T_ij_end=T_ij_end) def transverse_distance_start_stop(self, z_start, z_stop, include_z_start=False): """ computes the transverse distance (T_ij) that is required by the ray-tracing between the starting redshift and the first deflector afterwards and the last deflector before the end of the ray-tracing. :param z_start: redshift of the start of the ray-tracing :param z_stop: stop of ray-tracing :param include_z_start: bool, i :return: T_ij_start, T_ij_end """ return self._multi_plane_base.transverse_distance_start_stop( z_start, z_stop, include_z_start) def arrival_time(self, theta_x, theta_y, kwargs_lens, check_convention=True): """ light travel time relative to a straight path through the coordinate (0,0) Negative sign means earlier arrival time :param theta_x: angle in x-direction on the image :param theta_y: angle in y-direction on the image :param kwargs_lens: lens model keyword argument list :return: travel time in unit of days """ dt_geo, dt_grav = self.geo_shapiro_delay( theta_x, theta_y, kwargs_lens, check_convention=check_convention) return dt_geo + dt_grav def geo_shapiro_delay(self, theta_x, theta_y, kwargs_lens, check_convention=True): """ geometric and Shapiro (gravitational) light travel time relative to a straight path through the coordinate (0,0) Negative sign means earlier arrival time :param theta_x: angle in x-direction on the image :param theta_y: angle in y-direction on the image :param kwargs_lens: lens model keyword argument list :param check_convention: boolean, if True goes through the lens model list and checks whether the positional conventions are satisfied. :return: geometric delay, gravitational delay [days] """ if check_convention and not self.ignore_observed_positions: kwargs_lens = self._convention(kwargs_lens) return self._multi_plane_base.geo_shapiro_delay( theta_x, theta_y, kwargs_lens, z_stop=self._z_source, T_z_stop=self._T_z_source, T_ij_end=self._T_ij_stop) def alpha(self, theta_x, theta_y, kwargs_lens, check_convention=True, k=None): """ reduced deflection angle :param theta_x: angle in x-direction :param theta_y: angle in y-direction :param kwargs_lens: lens model kwargs :param check_convention: flag to check the image position convention (leave this alone) :return: deflection angles in x and y directions """ self._check_raise(k=k) beta_x, beta_y = self.ray_shooting(theta_x, theta_y, kwargs_lens, check_convention=check_convention) alpha_x = theta_x - beta_x alpha_y = theta_y - beta_y return alpha_x, alpha_y def hessian(self, theta_x, theta_y, kwargs_lens, k=None, diff=0.00000001, check_convention=True): """ computes the hessian components f_xx, f_yy, f_xy from f_x and f_y with numerical differentiation :param theta_x: x-position (preferentially arcsec) :type theta_x: numpy array :param theta_y: y-position (preferentially arcsec) :type theta_y: numpy array :param kwargs_lens: list of keyword arguments of lens model parameters matching the lens model classes :param diff: numerical differential step (float) :param check_convention: boolean, if True goes through the lens model list and checks whether the positional conventions are satisfied. :return: f_xx, f_xy, f_yx, f_yy """ self._check_raise(k=k) if check_convention and not self.ignore_observed_positions: kwargs_lens = self._convention(kwargs_lens) alpha_ra, alpha_dec = self.alpha(theta_x, theta_y, kwargs_lens, check_convention=False) alpha_ra_dx, alpha_dec_dx = self.alpha(theta_x + diff, theta_y, kwargs_lens, check_convention=False) alpha_ra_dy, alpha_dec_dy = self.alpha(theta_x, theta_y + diff, kwargs_lens, check_convention=False) dalpha_rara = (alpha_ra_dx - alpha_ra) / diff dalpha_radec = (alpha_ra_dy - alpha_ra) / diff dalpha_decra = (alpha_dec_dx - alpha_dec) / diff dalpha_decdec = (alpha_dec_dy - alpha_dec) / diff f_xx = dalpha_rara f_yy = dalpha_decdec f_xy = dalpha_radec f_yx = dalpha_decra return f_xx, f_xy, f_yx, f_yy def co_moving2angle_source(self, x, y): """ special case of the co_moving2angle definition at the source redshift :param x: co-moving distance :param y: co-moving distance :return: angles on the sky at the nominal source plane """ T_z = self._T_z_source theta_x = x / T_z theta_y = y / T_z return theta_x, theta_y def set_static(self, kwargs): """ :param kwargs: lens model keyword argument list :return: lens model keyword argument list with positional parameters all in flat sky coordinates """ kwargs = self.observed2flat_convention(kwargs) self.ignore_observed_positions = True return self._multi_plane_base.set_static(kwargs) def set_dynamic(self): """ :return: """ self.ignore_observed_positions = False self._multi_plane_base.set_dynamic() @staticmethod def _check_raise(k=None): """ checks whether no option to select a specific subset of deflector models is selected, as this feature is not yet supported in multi-plane :param k: parameter that optionally indicates a sub-set of lens models being executed for single plane :return: None, optional raise """ if k is not None: raise ValueError( 'no specific selection of a subset of lens models supported in multi-plane mode. Please' 'use single plane mode or generate new instance of LensModel of the subset of profiles.' )