Coverage for src/galsbi/ucat/galaxy_population_models/galaxy_shape.py: 87%
95 statements
« prev ^ index » next coverage.py v7.6.10, created at 2025-01-10 11:12 +0000
« prev ^ index » next coverage.py v7.6.10, created at 2025-01-10 11:12 +0000
1# Copyright (C) 2018 ETH Zurich, Institute for Particle Physics and Astrophysics
3"""
4Created 2021
5author: Tomasz Kacprzak
6"""
8from warnings import warn
10import numpy as np
11from scipy.stats import beta
13LOG_0_2539 = np.log(0.2539)
16def distortion_to_shear(distortion_x):
17 q_sq = (1 - distortion_x) / (1 + distortion_x)
18 q = np.sqrt(q_sq)
19 shear_x = (1 - q) / (1 + q)
20 return shear_x
23##########################################################
24#
25# Sample galaxy ellipticities
26#
27##########################################################
30def sample_ellipticities_gaussian(numgalaxies, e1_mean, e2_mean, e1_sigma, e2_sigma):
31 """
32 Sample Gaussian distributions for the intrinsic e1 and e2 while enforcing that
33 e1**2 + e2**2 <= 1
35 :param numgalaxies: number of galaxies, i.e. number of samples
36 :param par: ctx.parameters; part of ctx containing parameters
37 :return: e1 values
38 :return: e2 values
39 """
41 gal_e1 = np.ones(numgalaxies)
42 gal_e2 = np.ones(numgalaxies)
44 while np.any(gal_e1**2 + gal_e2**2 >= 1.0):
45 index = gal_e1**2 + gal_e2**2 >= 1.0
47 if np.any(index):
48 gal_e1[index] = np.random.normal(e1_mean, e1_sigma, size=np.sum(index))
49 gal_e2[index] = np.random.normal(e2_mean, e2_sigma, size=np.sum(index))
51 return gal_e1, gal_e2
54def pe_disc(ngal, log_a=LOG_0_2539, emax=0.8, emin=0.0256, pow_alpha=1):
55 """
56 From miller2013.
57 """
59 e_distortion = np.linspace(0, 1, 100000)
60 e_shear = distortion_to_shear(e_distortion)
61 pe = (
62 (e_shear**pow_alpha)
63 * (1 - np.exp((e_shear - emax) / np.exp(log_a)))
64 / (1.0 + e_shear)
65 / np.sqrt(e_shear**2 + emin**2)
66 )
67 pe[e_shear > 1] = 0
68 pe[pe < 0] = 0
69 pe = pe / np.sum(pe)
70 pep = np.random.choice(a=e_distortion, size=ngal, p=pe)
71 ea = pep * np.exp(1j * np.random.uniform(0.0, 2 * np.pi, len(pep)))
72 e1, e2 = ea.real, ea.imag
73 return e1, e2
76def pe_bulge(ngal, b=2.368, c=6.691):
77 """
78 From miller2013.
79 """
81 e_distortion = np.linspace(0, 1, 100000)
82 e_shear = distortion_to_shear(e_distortion)
83 pe = e_shear * np.exp(-b * e_shear - c * e_shear**2)
84 pe[e_shear > 1] = 0
85 pe[pe < 0] = 0
86 pe = pe / np.sum(pe)
87 pep = np.random.choice(a=e_distortion, size=ngal, p=pe)
88 ea = pep * np.exp(1j * np.random.uniform(0.0, 2 * np.pi, len(pep)))
89 e1, e2 = ea.real, ea.imag
90 return e1, e2
93def sample_ellipticities_beta(n_gal, par):
94 grid_e = np.linspace(0, 1, 100000)
95 beta_a = par.ell_beta_ab_sum * (par.ell_beta_ab_ratio)
96 beta_b = par.ell_beta_ab_sum * (1 - par.ell_beta_ab_ratio)
97 rv = beta(beta_a, beta_b)
98 pe = (grid_e / par.ell_beta_emax) ** 0.2 * rv.pdf(grid_e / par.ell_beta_emax)
99 pe[~np.isfinite(pe)] = 0
100 pe = pe / np.sum(pe)
101 pep = np.random.choice(a=grid_e, size=n_gal, p=pe)
102 ea = pep * np.exp(1j * np.random.uniform(0.0, 2 * np.pi, len(pep)))
103 e1, e2 = ea.real, ea.imag
104 return e1, e2
107def sample_ellipticities_beta_mode(
108 n_gal, ell_beta_ab_sum, ell_beta_mode, ell_beta_emax
109):
110 beta_a = ell_beta_ab_sum * ell_beta_mode - 2 * ell_beta_mode + 1
111 beta_b = ell_beta_ab_sum - beta_a
112 pep = beta.rvs(beta_a, beta_b, scale=ell_beta_emax, size=n_gal)
113 ea = pep * np.exp(1j * np.random.uniform(0.0, 2 * np.pi, len(pep)))
114 e1, e2 = ea.real, ea.imag
115 return e1, e2
118def sample_ellipticities_for_galaxy_type(n_gal, galaxy_type, par):
119 backwards_compatibility(par)
121 if par.ellipticity_sampling_method == "gaussian":
122 int_e1, int_e2 = sample_ellipticities_gaussian(
123 numgalaxies=n_gal,
124 e1_mean=par.e1_mean,
125 e2_mean=par.e2_mean,
126 e1_sigma=par.e1_sigma,
127 e2_sigma=par.e2_sigma,
128 )
130 elif par.ellipticity_sampling_method == "gaussian_blue_red":
131 int_e1, int_e2 = sample_ellipticities_gaussian(
132 numgalaxies=n_gal,
133 e1_mean=getattr(par, f"e1_mean_{galaxy_type}"),
134 e2_mean=getattr(par, f"e2_mean_{galaxy_type}"),
135 e1_sigma=getattr(par, f"ell_sigma_{galaxy_type}"),
136 e2_sigma=getattr(par, f"ell_sigma_{galaxy_type}"),
137 )
139 elif par.ellipticity_sampling_method == "blue_red_miller2013":
140 if galaxy_type == "blue":
141 int_e1, int_e2 = pe_disc(
142 ngal=n_gal,
143 log_a=par.ell_disc_log_a,
144 emin=par.ell_disc_min_e,
145 pow_alpha=par.ell_disc_pow_alpha,
146 )
148 elif galaxy_type == "red":
149 int_e1, int_e2 = pe_bulge(ngal=n_gal, b=par.ell_bulge_b)
151 else:
152 raise ValueError(
153 f"galaxy type {galaxy_type} not supported for"
154 " par.ellipticity_sampling_method=blue_red_miller2013"
155 )
157 elif par.ellipticity_sampling_method == "beta_ratio":
158 int_e1, int_e2 = sample_ellipticities_beta(n_gal=n_gal, par=par)
160 elif par.ellipticity_sampling_method == "beta_mode":
161 int_e1, int_e2 = sample_ellipticities_beta_mode(
162 n_gal=n_gal,
163 ell_beta_ab_sum=par.ell_beta_ab_sum,
164 ell_beta_mode=par.ell_beta_mode,
165 ell_beta_emax=par.ell_beta_emax,
166 )
168 elif par.ellipticity_sampling_method == "beta_mode_red_blue":
169 int_e1, int_e2 = sample_ellipticities_beta_mode(
170 n_gal=n_gal,
171 ell_beta_ab_sum=getattr(par, f"ell_beta_ab_sum_{galaxy_type}"),
172 ell_beta_mode=getattr(par, f"ell_beta_mode_{galaxy_type}"),
173 ell_beta_emax=par.ell_beta_emax,
174 )
176 else:
177 raise ValueError(
178 f"unknown ellipticity_sampling_method {par.ellipticity_sampling_method}"
179 )
181 return int_e1, int_e2
184def backwards_compatibility(par):
185 if par.ellipticity_sampling_method == "default":
186 warn(
187 DeprecationWarning(
188 "ellipticity_sampling_method=default is deprecated, "
189 "using gaussian instead"
190 ),
191 stacklevel=1,
192 )
193 par.ellipticity_sampling_method = "gaussian"
195 if par.ellipticity_sampling_method == "blue_red":
196 warn(
197 DeprecationWarning(
198 "ellipticity_sampling_method=blue_red is deprecated, "
199 "using gaussian_blue_red instead"
200 ),
201 stacklevel=1,
202 )
203 par.ellipticity_sampling_method = "gaussian_blue_red"
205 if par.ellipticity_sampling_method == "beta_function":
206 warn(
207 DeprecationWarning(
208 "ellipticity_sampling_method=beta_function is deprecated, "
209 "using beta_ratio instead"
210 ),
211 stacklevel=1,
212 )
213 par.ellipticity_sampling_method = "beta_ratio"
215 if par.ellipticity_sampling_method == "beta_function_mode":
216 warn(
217 DeprecationWarning(
218 "ellipticity_sampling_method=beta_function_mode is deprecated, "
219 "using beta_mode instead"
220 ),
221 stacklevel=1,
222 )
223 par.ellipticity_sampling_method = "beta_mode"
225 if par.ellipticity_sampling_method == "beta_function_mode_red_blue":
226 warn(
227 DeprecationWarning(
228 "ellipticity_sampling_method=beta_function_mode_red_blue is "
229 "deprecated, using beta_mode_red_blue instead"
230 ),
231 stacklevel=1,
232 )
233 par.ellipticity_sampling_method = "beta_mode_red_blue"