forked from farr/SelectionExample
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathschechter.stan
87 lines (68 loc) · 2.49 KB
/
schechter.stan
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
functions {
real schechter_lpdf(vector L, real alpha, real Lstar) {
return sum(alpha*log(L) - L/Lstar - (alpha+1)*log(Lstar) - lgamma(alpha+1));
}
}
data {
int Nobs;
vector[Nobs] Lobs;
int NNobs_max;
real Funcert; /* Fractional flux uncertainty */
real zmax; /* Maximum redshift. */
real Fth; /* Threshold flux */
}
parameters {
/* Expected number of sources out to zmax. */
real<lower=0> Lambda_total;
real<lower=0,upper=1> f_phys;
/* Power law at low luminosity. */
real<lower=-1> alpha;
/* Turnover to exponential decay. */
real<lower=0> Lstar;
/* True luminosity inferred for the observed systems. */
vector<lower=0>[Nobs] Ltrue;
/* Next parameters refer to the un-observed systems. */
/* True luminosity of the (possibly) un-observed systems; making
this vector `positive_ordered` makes the un-observed systems
distinguishable. */
positive_ordered[NNobs_max] Ltrue_nobs;
/* True redshift of (possibly) unobserved systems. */
vector<lower=0,upper=zmax>[NNobs_max] ztrue_nobs;
/* To be non-observed, we must have a flux smaller than the flux
limit. */
vector<lower=0,upper=Fth>[NNobs_max] flux_nobs;
}
transformed parameters {
real Lambda = f_phys*Lambda_total;
real Lambda0 = (1-f_phys)*Lambda_total;
}
model {
/* Priors. */
/* Lambda and Lambda0 are related to f_phys and Lambda_total, but we need a Jacobian:
d(Lambda, Lambda0)/d(Lambda_total, f_phys) = -2 Lambda_total
*/
Lambda ~ normal(0.0, 200.0);
Lambda0 ~ normal(0.0, NNobs_max);
target += log(Lambda_total); /* Proportional to log(jacobian). */
alpha ~ normal(0.0, 1.0);
Lstar ~ normal(0.0, 2.0);
/* Observed systems (note that P_det == 1 for these systems, since
their flux is above the limit). */
Ltrue ~ schechter(alpha, Lstar);
target += Nobs*log(Lambda);
Lobs ~ lognormal(log(Ltrue), Funcert);
/* Non-observed systems are a mix of *physical* systems (counted by
Lambda), whose flux follows from their luminosity and
*non-physical* systems (counted by Lambda0) whose flux is not
tied to their luminosity at all, and instead distributed flat up
to the flux threshold. */
Ltrue_nobs ~ schechter(alpha, Lstar);
/* Redshifts are flat */
for (i in 1:NNobs_max) {
real log_ex_flux = log(Ltrue_nobs[i]) - log(4.0*pi()) - 2.0*log(ztrue_nobs[i]);
target += log_sum_exp(log(Lambda0) - log(Fth),
log(Lambda) + lognormal_lpdf(flux_nobs[i] | log_ex_flux, Funcert));
}
/* Poisson normalisation */
target += -Lambda - Lambda0;
}