Fitting a model to data with both x and y errors with Bilby
Usually when we fit a model to data with a Gaussian Likelihood we assume that we know x values exactly. This is almost never the case. Here we show how to fit a model with errors in both x and y.
[1]:
import bilby
import matplotlib.pyplot as plt
import numpy as np
from bilby.core.utils.random import seed, rng
#sets seed of bilby's generator "rng" to "123"
seed(123)
%matplotlib inline
Simulate data
First we create the data and plot it
[2]:
# define our model, a line
def model(x, m, c, **kwargs):
y = m * x + c
return y
# make a function to create and plot our data
def make_data(points, m, c, xerr, yerr, seed):
xtrue = np.linspace(0, 100, points)
ytrue = model(x=xtrue, m=m, c=c)
xerr_vals = xerr * rng.standard_normal(points)
yerr_vals = yerr * rng.standard_normal(points)
xobs = xtrue + xerr_vals
yobs = ytrue + yerr_vals
plt.errorbar(xobs, yobs, xerr=xerr, yerr=yerr, fmt="x")
plt.errorbar(xtrue, ytrue, yerr=yerr, color="black", alpha=0.5)
plt.xlim(0, 100)
plt.show()
plt.close()
data = {
"xtrue": xtrue,
"ytrue": ytrue,
"xobs": xobs,
"yobs": yobs,
"xerr": xerr,
"yerr": yerr,
}
return data
data = make_data(points=30, m=5, c=10, xerr=5, yerr=5, seed=123)
Define our prior and sampler settings
Now lets set up the prior and bilby output directory/sampler settings
[3]:
# setting up bilby priors
priors = dict(
m=bilby.core.prior.Uniform(0, 30, "m"), c=bilby.core.prior.Uniform(0, 30, "c")
)
sampler_kwargs = dict(priors=priors, sampler="bilby_mcmc", nsamples=1000, printdt=5, outdir="outdir", verbose=False, clean=True)
Fit with exactly known x-values
Our first step is to recover the straight line using a simple Gaussian Likelihood that only takes into account the y errors. Under the assumption we know x exactly. In this case, we pass in xtrue for x
[4]:
known_x = bilby.core.likelihood.GaussianLikelihood(
x=data["xtrue"], y=data["yobs"], func=model, sigma=data["yerr"]
)
result_known_x = bilby.run_sampler(
likelihood=known_x,
label="known_x",
**sampler_kwargs,
)
16:32 bilby INFO : Running for label 'known_x', output will be saved to 'outdir'
16:32 bilby INFO : Analysis priors:
16:32 bilby INFO : m=Uniform(minimum=0, maximum=30, name='m', latex_label='m', unit=None, boundary=None)
16:32 bilby INFO : c=Uniform(minimum=0, maximum=30, name='c', latex_label='c', unit=None, boundary=None)
16:32 bilby INFO : Analysis likelihood class: <class 'bilby.core.likelihood.GaussianLikelihood'>
16:32 bilby INFO : Analysis likelihood noise evidence: nan
16:32 bilby INFO : Single likelihood evaluation took 4.859e-05 s
16:32 bilby INFO : Using sampler Bilby_MCMC with kwargs {'nsamples': 1000, 'nensemble': 1, 'pt_ensemble': False, 'ntemps': 1, 'Tmax': None, 'Tmax_from_SNR': 20, 'initial_betas': None, 'adapt': True, 'adapt_t0': 100, 'adapt_nu': 10, 'pt_rejection_sample': False, 'burn_in_nact': 10, 'thin_by_nact': 1, 'fixed_discard': 0, 'autocorr_c': 5, 'L1steps': 100, 'L2steps': 3, 'printdt': 5, 'check_point_delta_t': 1800, 'min_tau': 1, 'proposal_cycle': 'default', 'stop_after_convergence': False, 'fixed_tau': None, 'tau_window': None, 'evidence_method': 'stepping_stone', 'initial_sample_method': 'prior', 'initial_sample_dict': None}
16:32 bilby INFO : Initializing BilbyPTMCMCSampler with:
Convergence settings: ConvergenceInputs(autocorr_c=5, burn_in_nact=10, thin_by_nact=1, fixed_discard=0, target_nsamples=1000, stop_after_convergence=False, L1steps=100, L2steps=3, min_tau=1, fixed_tau=None, tau_window=None)
Parallel-tempering settings: ParallelTemperingInputs(ntemps=1, nensemble=1, Tmax=None, Tmax_from_SNR=20, initial_betas=None, adapt=True, adapt_t0=100, adapt_nu=10, pt_ensemble=False)
proposal_cycle: default
pt_rejection_sample: False
16:32 bilby INFO : Setting parallel tempering inputs=ParallelTemperingInputs(ntemps=1, nensemble=1, Tmax=None, Tmax_from_SNR=20, initial_betas=None, adapt=True, adapt_t0=100, adapt_nu=10, pt_ensemble=False)
16:32 bilby INFO : Initializing BilbyPTMCMCSampler with:ntemps=1, nensemble=1, pt_ensemble=False, initial_betas=[1], initial_sample_method=prior, initial_sample_dict=None
16:32 bilby INFO : Using initial sample {'m': 9.290579764179022, 'c': 5.310327080197345}
16:32 bilby INFO : Using ProposalCycle:
AdaptiveGaussianProposal(acceptance_ratio:nan,n:0,scale:1,)
DifferentialEvolutionProposal(acceptance_ratio:nan,n:0,)
UniformProposal(acceptance_ratio:nan,n:0,)
KDEProposal(acceptance_ratio:nan,n:0,trained:0,)
FisherMatrixProposal(acceptance_ratio:nan,n:0,scale:1,)
GMMProposal(acceptance_ratio:nan,n:0,trained:0,)
16:32 bilby INFO : Setting convergence_inputs=ConvergenceInputs(autocorr_c=5, burn_in_nact=10, thin_by_nact=1, fixed_discard=0, target_nsamples=1000, stop_after_convergence=False, L1steps=100, L2steps=3, min_tau=1, fixed_tau=None, tau_window=None)
16:32 bilby INFO : Drawing 1000 samples
16:32 bilby INFO : Checkpoint every check_point_delta_t=1800s
16:32 bilby INFO : Print update every printdt=5s
16:32 bilby INFO : Reached convergence: exiting sampling
16:32 bilby INFO : Checkpoint start
16:32 bilby INFO : Written checkpoint file outdir/known_x_resume.pickle
16:32 bilby INFO : Zero-temperature proposals:
16:32 bilby INFO : AdaptiveGaussianProposal(acceptance_ratio:0.23,n:3.4e+04,scale:0.0028,)
16:32 bilby INFO : DifferentialEvolutionProposal(acceptance_ratio:0.48,n:3.5e+04,)
16:32 bilby INFO : UniformProposal(acceptance_ratio:1,n:1.5e+03,)
16:32 bilby INFO : KDEProposal(acceptance_ratio:0.00012,n:3.4e+04,trained:0,)
16:32 bilby INFO : FisherMatrixProposal(acceptance_ratio:0.55,n:3.6e+04,scale:1,)
16:32 bilby INFO : GMMProposal(acceptance_ratio:6.1e-05,n:3.3e+04,trained:0,)
16:32 bilby INFO : Current taus={'m': 1.2, 'c': 1.2}
16:32 bilby INFO : Creating diagnostic plots
16:32 bilby INFO : Checkpoint finished
16:32 bilby INFO : Sampling time: 0:00:15.011096
16:32 bilby INFO : Summary of results:
nsamples: 1628
ln_noise_evidence: nan
ln_evidence: nan +/- nan
ln_bayes_factor: nan +/- nan
[5]:
_ = result_known_x.plot_corner(truth=dict(m=5, c=10), titles=True, save=False)
plt.show()
plt.close()
Fit with unmodeled uncertainty in the x-values
As expected this is easy to recover and the sampler does a good job. However this was made too easy - by passing in the ‘true’ values of x. Lets see what happens when we pass in the observed values of x
[6]:
incorrect_x = bilby.core.likelihood.GaussianLikelihood(
x=data["xobs"], y=data["yobs"], func=model, sigma=data["yerr"]
)
result_incorrect_x = bilby.run_sampler(
likelihood=incorrect_x,
label="incorrect_x",
**sampler_kwargs,
)
16:32 bilby INFO : Running for label 'incorrect_x', output will be saved to 'outdir'
16:33 bilby INFO : Analysis priors:
16:33 bilby INFO : m=Uniform(minimum=0, maximum=30, name='m', latex_label='m', unit=None, boundary=None)
16:33 bilby INFO : c=Uniform(minimum=0, maximum=30, name='c', latex_label='c', unit=None, boundary=None)
16:33 bilby INFO : Analysis likelihood class: <class 'bilby.core.likelihood.GaussianLikelihood'>
16:33 bilby INFO : Analysis likelihood noise evidence: nan
16:33 bilby INFO : Single likelihood evaluation took 6.531e-05 s
16:33 bilby INFO : Using sampler Bilby_MCMC with kwargs {'nsamples': 1000, 'nensemble': 1, 'pt_ensemble': False, 'ntemps': 1, 'Tmax': None, 'Tmax_from_SNR': 20, 'initial_betas': None, 'adapt': True, 'adapt_t0': 100, 'adapt_nu': 10, 'pt_rejection_sample': False, 'burn_in_nact': 10, 'thin_by_nact': 1, 'fixed_discard': 0, 'autocorr_c': 5, 'L1steps': 100, 'L2steps': 3, 'printdt': 5, 'check_point_delta_t': 1800, 'min_tau': 1, 'proposal_cycle': 'default', 'stop_after_convergence': False, 'fixed_tau': None, 'tau_window': None, 'evidence_method': 'stepping_stone', 'initial_sample_method': 'prior', 'initial_sample_dict': None}
16:33 bilby INFO : Initializing BilbyPTMCMCSampler with:
Convergence settings: ConvergenceInputs(autocorr_c=5, burn_in_nact=10, thin_by_nact=1, fixed_discard=0, target_nsamples=1000, stop_after_convergence=False, L1steps=100, L2steps=3, min_tau=1, fixed_tau=None, tau_window=None)
Parallel-tempering settings: ParallelTemperingInputs(ntemps=1, nensemble=1, Tmax=None, Tmax_from_SNR=20, initial_betas=None, adapt=True, adapt_t0=100, adapt_nu=10, pt_ensemble=False)
proposal_cycle: default
pt_rejection_sample: False
16:33 bilby INFO : Setting parallel tempering inputs=ParallelTemperingInputs(ntemps=1, nensemble=1, Tmax=None, Tmax_from_SNR=20, initial_betas=None, adapt=True, adapt_t0=100, adapt_nu=10, pt_ensemble=False)
16:33 bilby INFO : Initializing BilbyPTMCMCSampler with:ntemps=1, nensemble=1, pt_ensemble=False, initial_betas=[1], initial_sample_method=prior, initial_sample_dict=None
16:33 bilby INFO : Using initial sample {'m': 0.05634655217674589, 'c': 7.764712753819197}
16:33 bilby INFO : Using ProposalCycle:
AdaptiveGaussianProposal(acceptance_ratio:nan,n:0,scale:1,)
DifferentialEvolutionProposal(acceptance_ratio:nan,n:0,)
UniformProposal(acceptance_ratio:nan,n:0,)
KDEProposal(acceptance_ratio:nan,n:0,trained:0,)
FisherMatrixProposal(acceptance_ratio:nan,n:0,scale:1,)
GMMProposal(acceptance_ratio:nan,n:0,trained:0,)
16:33 bilby INFO : Setting convergence_inputs=ConvergenceInputs(autocorr_c=5, burn_in_nact=10, thin_by_nact=1, fixed_discard=0, target_nsamples=1000, stop_after_convergence=False, L1steps=100, L2steps=3, min_tau=1, fixed_tau=None, tau_window=None)
16:33 bilby INFO : Drawing 1000 samples
16:33 bilby INFO : Checkpoint every check_point_delta_t=1800s
16:33 bilby INFO : Print update every printdt=5s
16:33 bilby INFO : Reached convergence: exiting sampling
16:33 bilby INFO : Checkpoint start
16:33 bilby INFO : Written checkpoint file outdir/incorrect_x_resume.pickle
16:33 bilby INFO : Zero-temperature proposals:
16:33 bilby INFO : AdaptiveGaussianProposal(acceptance_ratio:0.23,n:2.7e+04,scale:0.00046,)
16:33 bilby INFO : DifferentialEvolutionProposal(acceptance_ratio:0.46,n:2.3e+04,)
16:33 bilby INFO : UniformProposal(acceptance_ratio:1,n:1.1e+03,)
16:33 bilby INFO : KDEProposal(acceptance_ratio:0.0001,n:1.9e+04,trained:0,)
16:33 bilby INFO : FisherMatrixProposal(acceptance_ratio:0.56,n:2.4e+04,scale:1,)
16:33 bilby INFO : GMMProposal(acceptance_ratio:4.5e-05,n:2.2e+04,trained:0,)
16:33 bilby INFO : Current taus={'m': 1, 'c': 1}
16:33 bilby INFO : Creating diagnostic plots
16:33 bilby INFO : Checkpoint finished
16:33 bilby INFO : Sampling time: 0:00:10.001872
16:33 bilby INFO : Summary of results:
nsamples: 1028
ln_noise_evidence: nan
ln_evidence: nan +/- nan
ln_bayes_factor: nan +/- nan
[7]:
_ = result_incorrect_x.plot_corner(truth=dict(m=5, c=10), titles=True, save=False)
plt.show()
plt.close()
Fit with modeled uncertainty in x-values
This is not good as there is unmodelled uncertainty in our x
values. Getting around this requires marginalisation of the true x values or sampling over them. See discussion in section 7 of https://arxiv.org/pdf/1008.4686.pdf.
For this, we will have to define a new likelihood class. By subclassing the base bilby.core.likelihood.Likelihood
class we can do this fairly simply.
[8]:
class GaussianLikelihoodUncertainX(bilby.core.likelihood.Likelihood):
def __init__(self, xobs, yobs, xerr, yerr, function):
"""
Parameters
----------
xobs, yobs: array_like
The data to analyse
xerr, yerr: array_like
The standard deviation of the noise
function:
The python function to fit to the data
"""
super(GaussianLikelihoodUncertainX, self).__init__(dict())
self.xobs = xobs
self.yobs = yobs
self.yerr = yerr
self.xerr = xerr
self.function = function
def log_likelihood(self):
variance = (self.xerr * self.parameters["m"]) ** 2 + self.yerr**2
model_y = self.function(self.xobs, **self.parameters)
residual = self.yobs - model_y
ll = -0.5 * np.sum(residual**2 / variance + np.log(variance))
return -0.5 * np.sum(residual**2 / variance + np.log(variance))
[9]:
gaussian_unknown_x = GaussianLikelihoodUncertainX(
xobs=data["xobs"],
yobs=data["yobs"],
xerr=data["xerr"],
yerr=data["yerr"],
function=model,
)
result_unknown_x = bilby.run_sampler(
likelihood=gaussian_unknown_x,
label="unknown_x",
**sampler_kwargs,
)
16:33 bilby INFO : Running for label 'unknown_x', output will be saved to 'outdir'
16:33 bilby INFO : Analysis priors:
16:33 bilby INFO : m=Uniform(minimum=0, maximum=30, name='m', latex_label='m', unit=None, boundary=None)
16:33 bilby INFO : c=Uniform(minimum=0, maximum=30, name='c', latex_label='c', unit=None, boundary=None)
16:33 bilby INFO : Analysis likelihood class: <class '__main__.GaussianLikelihoodUncertainX'>
16:33 bilby INFO : Analysis likelihood noise evidence: nan
16:33 bilby INFO : Single likelihood evaluation took 5.556e-05 s
16:33 bilby INFO : Using sampler Bilby_MCMC with kwargs {'nsamples': 1000, 'nensemble': 1, 'pt_ensemble': False, 'ntemps': 1, 'Tmax': None, 'Tmax_from_SNR': 20, 'initial_betas': None, 'adapt': True, 'adapt_t0': 100, 'adapt_nu': 10, 'pt_rejection_sample': False, 'burn_in_nact': 10, 'thin_by_nact': 1, 'fixed_discard': 0, 'autocorr_c': 5, 'L1steps': 100, 'L2steps': 3, 'printdt': 5, 'check_point_delta_t': 1800, 'min_tau': 1, 'proposal_cycle': 'default', 'stop_after_convergence': False, 'fixed_tau': None, 'tau_window': None, 'evidence_method': 'stepping_stone', 'initial_sample_method': 'prior', 'initial_sample_dict': None}
16:33 bilby INFO : Initializing BilbyPTMCMCSampler with:
Convergence settings: ConvergenceInputs(autocorr_c=5, burn_in_nact=10, thin_by_nact=1, fixed_discard=0, target_nsamples=1000, stop_after_convergence=False, L1steps=100, L2steps=3, min_tau=1, fixed_tau=None, tau_window=None)
Parallel-tempering settings: ParallelTemperingInputs(ntemps=1, nensemble=1, Tmax=None, Tmax_from_SNR=20, initial_betas=None, adapt=True, adapt_t0=100, adapt_nu=10, pt_ensemble=False)
proposal_cycle: default
pt_rejection_sample: False
16:33 bilby INFO : Setting parallel tempering inputs=ParallelTemperingInputs(ntemps=1, nensemble=1, Tmax=None, Tmax_from_SNR=20, initial_betas=None, adapt=True, adapt_t0=100, adapt_nu=10, pt_ensemble=False)
16:33 bilby INFO : Initializing BilbyPTMCMCSampler with:ntemps=1, nensemble=1, pt_ensemble=False, initial_betas=[1], initial_sample_method=prior, initial_sample_dict=None
16:33 bilby INFO : Using initial sample {'m': 17.415022274341336, 'c': 5.487143937464902}
16:33 bilby INFO : Using ProposalCycle:
AdaptiveGaussianProposal(acceptance_ratio:nan,n:0,scale:1,)
DifferentialEvolutionProposal(acceptance_ratio:nan,n:0,)
UniformProposal(acceptance_ratio:nan,n:0,)
KDEProposal(acceptance_ratio:nan,n:0,trained:0,)
FisherMatrixProposal(acceptance_ratio:nan,n:0,scale:1,)
GMMProposal(acceptance_ratio:nan,n:0,trained:0,)
16:33 bilby INFO : Setting convergence_inputs=ConvergenceInputs(autocorr_c=5, burn_in_nact=10, thin_by_nact=1, fixed_discard=0, target_nsamples=1000, stop_after_convergence=False, L1steps=100, L2steps=3, min_tau=1, fixed_tau=None, tau_window=None)
16:33 bilby INFO : Drawing 1000 samples
16:33 bilby INFO : Checkpoint every check_point_delta_t=1800s
16:33 bilby INFO : Print update every printdt=5s
16:33 bilby INFO : Reached convergence: exiting sampling
16:33 bilby INFO : Checkpoint start
16:33 bilby INFO : Written checkpoint file outdir/unknown_x_resume.pickle
16:33 bilby INFO : Zero-temperature proposals:
16:33 bilby INFO : AdaptiveGaussianProposal(acceptance_ratio:0.23,n:2.4e+04,scale:0.018,)
16:33 bilby INFO : DifferentialEvolutionProposal(acceptance_ratio:0.46,n:2.5e+04,)
16:33 bilby INFO : UniformProposal(acceptance_ratio:1,n:5.9e+02,)
16:33 bilby INFO : KDEProposal(acceptance_ratio:0.00086,n:2.4e+04,trained:0,)
16:33 bilby INFO : FisherMatrixProposal(acceptance_ratio:0.51,n:2.4e+04,scale:1,)
16:33 bilby INFO : GMMProposal(acceptance_ratio:0.0011,n:2.1e+04,trained:0,)
16:33 bilby INFO : Current taus={'m': 1, 'c': 1.0}
16:33 bilby INFO : Creating diagnostic plots
16:33 bilby INFO : Checkpoint finished
16:33 bilby INFO : Sampling time: 0:00:10.004996
16:33 bilby INFO : Summary of results:
nsamples: 1084
ln_noise_evidence: nan
ln_evidence: nan +/- nan
ln_bayes_factor: nan +/- nan
[10]:
_ = result_unknown_x.plot_corner(truth=dict(m=5, c=10), titles=True, save=False)
plt.show()
plt.close()
Success! The inferred posterior is consistent with the true values.