.. DO NOT EDIT. .. THIS FILE WAS AUTOMATICALLY GENERATED BY SPHINX-GALLERY. .. TO MAKE CHANGES, EDIT THE SOURCE PYTHON FILE: .. "examples/spec_with_photo.py" .. LINE NUMBERS ARE GIVEN BELOW. .. only:: html .. note:: :class: sphx-glr-download-link-note :ref:`Go to the end ` to download the full example code. .. rst-class:: sphx-glr-example-title .. _sphx_glr_examples_spec_with_photo.py: NUTS analysis of the high-resolution emission spectrum with photometric information ==================================================================================== This script performs a Bayesian analysis of the high-resolution emission spectrum with photometric information. .. GENERATED FROM PYTHON SOURCE LINES 8-22 .. code-block:: Python # from jax import config # config.update("jax_enable_x64", True) import numpy as np import jax.numpy as jnp import matplotlib.pyplot as plt from exojax.spec.specop import SopPhoto from exojax.test.emulate_spec import sample_emission_spectrum from exojax.utils.grids import wavenumber_grid from exojax.utils.instfunc import resolution_to_gaussian_std from jax import jit .. GENERATED FROM PYTHON SOURCE LINES 23-30 loads test spectrum data and sets prior knowledge, used in Paper I (Kawahara et al. 2022) ------------------------------------------------------------------------------------------- Here, we sets the test spectrum data of Luhman 16A, which is used in Paper I (Kawahara et al. 2022). Kmag and Kmag_err are the K-band magnitude and its error, respectively. we also sets the instrumental resolution and the prior knowledge of the planet. .. GENERATED FROM PYTHON SOURCE LINES 30-70 .. code-block:: Python nu_grid_obs, f_obs, f_obs_err, Kmag, Kmag_err, filter_id = sample_emission_spectrum() fig = plt.figure(figsize=(20, 4)) plt.plot(nu_grid_obs, f_obs, ".", c="k", label="data", alpha=0.3) plt.ylim(0.4, 1.3) plt.legend() plt.savefig("obsspec.png") distance = 1.996 # pc Bedin et al. 2023 # instrument settings Rinst = 100000.0 # instrumental spectral resolution beta_inst = resolution_to_gaussian_std(Rinst) # normalization (just to use the spectrum to be around 1) Fref = 2.0e-15 # wavenumber grid setting for the forward model nu_min = np.min(nu_grid_obs) nu_max = np.max(nu_grid_obs) margin = 1.0 ngrid = 6000 nu_grid_spec, wav, res = wavenumber_grid( nu_min - margin, nu_max + margin, ngrid, xsmode="premodit" ) print("resolution = ", res) # ---- # photometry forward model setting sop_photo = SopPhoto(filter_id, up_resolution_factor=10) nu_grid_photo = sop_photo.nu_grid_filter print("len(nu_grid_photo) = ", len(nu_grid_photo)) #11082 print("len(nu_grid_spec) = ", len(nu_grid_spec)) #6000 # photometry model is more time-consuming than the spectroscopy model .. GENERATED FROM PYTHON SOURCE LINES 71-79 Molecular and CIA database settings -------------------------------------- Here, we set the molecular and CIA databases for the forward model. We use H2O, CH4, and CO for the molecular opacity and H2-H2 and H2-He for the CIA opacity. The molecular databases are from ExoMol (H2O and CO) and HITEMP (CH4), and the CIA databases are from the HITRAN CIA database. .. GENERATED FROM PYTHON SOURCE LINES 79-110 .. code-block:: Python # molecules/CIA database settings, uses nu_photo becuase it's wider than nu_grid_obs from exojax.spec.api import MdbExomol from exojax.spec.api import MdbHitemp from exojax.spec import contdb from exojax.spec import molinfo mdb_h2o = MdbExomol( ".database/H2O/1H2-16O/POKAZATEL", nurange=nu_grid_photo, gpu_transfer=False ) mdb_co = MdbExomol( ".database/CO/12C-16O/Li2015", nurange=nu_grid_photo, gpu_transfer=False ) mdb_ch4 = MdbHitemp(".database/CH4", nurange=nu_grid_photo, gpu_transfer=False) molmasses = jnp.array([mdb_h2o.molmass, mdb_ch4.molmass, mdb_co.molmass]) molmasses = jnp.array([mdb_h2o.molmass]) cdbH2H2 = contdb.CdbCIA(".database/H2-H2_2011.cia", nu_grid_photo) cdbH2He = contdb.CdbCIA(".database/H2-He_2011.cia", nu_grid_photo) molmassH2 = molinfo.molmass("H2") molmassHe = molinfo.molmass("He", db_HIT=False) # moelcular weight calculator def mean_molecular_weight(vmr, vmrH2, vmrHe): mmw = jnp.sum(vmr * molmasses) + vmrH2 * molmassH2 + vmrHe * molmassHe return mmw .. GENERATED FROM PYTHON SOURCE LINES 111-120 opacity calculators for spectroscopy -------------------------------------- Next, we define the opacity for the spectrum. Here, we use `OpaPremodit`` to generate `opa` of H2O, CO, and CH4 using their respective `mdb`. The `dit_grid_resolution` parameter determines the accuracy of the pressure width generation; in this case, we set it to a low accuracy of 1.0. Please check the required accuracy as needed. `auto_trange` is the valid temperature range for the opacity calculation. .. GENERATED FROM PYTHON SOURCE LINES 120-152 .. code-block:: Python from exojax.spec.opacalc import OpaPremodit from exojax.spec.opacont import OpaCIA trange = [500.0, 2500.0] dgres = 1.0 opa_spec_h2o = OpaPremodit( mdb_h2o, nu_grid_spec, auto_trange=trange, allow_32bit=True, dit_grid_resolution=dgres, ) opa_spec_co = OpaPremodit( mdb_co, nu_grid_spec, auto_trange=trange, allow_32bit=True, dit_grid_resolution=dgres, ) opa_spec_ch4 = OpaPremodit( mdb_ch4, nu_grid_spec, auto_trange=trange, allow_32bit=True, dit_grid_resolution=dgres, ) opa_spec_cia_H2H2 = OpaCIA(cdbH2H2, nu_grid_spec) opa_spec_cia_H2He = OpaCIA(cdbH2He, nu_grid_spec) .. GENERATED FROM PYTHON SOURCE LINES 153-158 opacity calculators for photometry -------------------------------------- We also define the opacity for the photometry. .. GENERATED FROM PYTHON SOURCE LINES 158-183 .. code-block:: Python opa_photo_h2o = OpaPremodit( mdb_h2o, nu_grid_photo, auto_trange=trange, allow_32bit=True, dit_grid_resolution=dgres, ) opa_photo_co = OpaPremodit( mdb_co, nu_grid_photo, auto_trange=trange, allow_32bit=True, dit_grid_resolution=dgres, ) opa_photo_ch4 = OpaPremodit( mdb_ch4, nu_grid_photo, auto_trange=trange, allow_32bit=True, dit_grid_resolution=dgres, ) opa_photo_cia_H2H2 = OpaCIA(cdbH2H2, nu_grid_photo) opa_photo_cia_H2He = OpaCIA(cdbH2He, nu_grid_photo) .. GENERATED FROM PYTHON SOURCE LINES 184-190 sets atmospheric radiative transfer model and spectral operators ----------------------------------------------------------------- Here, we set the atmospheric radiative transfer model and the spectral operators. we calculate the emission without scattering (pure absorption), so we construct `art` using `ArtEmisPure`. .. GENERATED FROM PYTHON SOURCE LINES 190-203 .. code-block:: Python from exojax.spec.atmrt import ArtEmisPure art = ArtEmisPure(pressure_btm=1.0e2, pressure_top=1.0e-4, nlayer=200) # does not set nu_grid because we use two types of nu_grid (for obs and photo) # Spectral Operators (planet rotation and instrumental profile) from exojax.spec.specop import SopRotation from exojax.spec.specop import SopInstProfile sop_rot = SopRotation(nu_grid_spec, vsini_max=100.0) sop_inst = SopInstProfile(nu_grid_spec, vrmax=100.0) .. GENERATED FROM PYTHON SOURCE LINES 204-209 defines the forward model for the spectroscopy --------------------------------------------------- Here, we define the forward model for the spectroscopy, `fspec``. .. GENERATED FROM PYTHON SOURCE LINES 209-276 .. code-block:: Python from exojax.utils.astrofunc import square_radius_from_mass_logg from exojax.utils.constants import RJ from exojax.utils.constants import pc # defines the calculation of the atmosphere def calc_atmosphere(T0, alpha, logg, Mp, logvmr): Tarr = art.powerlaw_temperature(T0, alpha) Parr = art.pressure gravity = 10**logg Rp2 = square_radius_from_mass_logg(Mp, logg) # VMRs and mean molecular weight vmr = jnp.power(10.0, jnp.array(logvmr)) vmrH2 = (1.0 - jnp.sum(vmr)) * 6.0 / 7.0 vmrHe = (1.0 - jnp.sum(vmr)) * 1.0 / 7.0 mmw = mean_molecular_weight(vmr, vmrH2, vmrHe) return Tarr, Parr, gravity, Rp2, vmr, vmrH2, vmrHe, mmw # defines a constant vmr profile def constant_vmr_profile(vmr): vmr_profile_h2o = art.constant_profile(vmr[0]) vmr_profile_co = art.constant_profile(vmr[1]) vmr_profile_ch4 = art.constant_profile(vmr[2]) return vmr_profile_h2o, vmr_profile_co, vmr_profile_ch4 # defines a spectral forward model @jit def fspec(T0, alpha, logg, Mp, logvmr, u1, u2, RV, vsini): Tarr, Parr, gravity, Rp2, vmr, vmrH2, vmrHe, mmw = calc_atmosphere( T0, alpha, logg, Mp, logvmr ) vmr_profile_h2o, vmr_profile_co, vmr_profile_ch4 = constant_vmr_profile(vmr) # cross sections for molecules xsm_h2o = opa_spec_h2o.xsmatrix(Tarr, Parr) xsm_co = opa_spec_co.xsmatrix(Tarr, Parr) xsm_ch4 = opa_spec_ch4.xsmatrix(Tarr, Parr) # sum of the opacity for molecules dtaum = art.opacity_profile_xs(xsm_h2o, vmr_profile_h2o, mmw, gravity) dtaum = dtaum + art.opacity_profile_xs(xsm_co, vmr_profile_co, mmw, gravity) dtaum = dtaum + art.opacity_profile_xs(xsm_ch4, vmr_profile_ch4, mmw, gravity) # CIAs dtauH2H2 = art.opacity_profile_cia( opa_spec_cia_H2H2.logacia_matrix(Tarr), Tarr, vmrH2, vmrH2, mmw, gravity ) dtauH2He = art.opacity_profile_cia( opa_spec_cia_H2He.logacia_matrix(Tarr), Tarr, vmrH2, vmrHe, mmw, gravity ) # sum of the opacity dtau_spec = dtaum + dtauH2H2 + dtauH2He F0 = art.run(dtau_spec, Tarr, nu_grid=nu_grid_spec) Frot = sop_rot.rigid_rotation(F0, vsini, u1, u2) Frot_inst = sop_inst.ipgauss(Frot, beta_inst) mu = sop_inst.sampling(Frot_inst, RV, nu_grid_obs) mu = mu * (Rp2 / distance**2) * (RJ / pc) ** 2 return mu / Fref .. GENERATED FROM PYTHON SOURCE LINES 277-282 defines photometry model ------------------------------ We also define the photometry model, `fphoto`. .. GENERATED FROM PYTHON SOURCE LINES 282-318 .. code-block:: Python @jit def fphoto(T0, alpha, logg, Mp, logvmr): Tarr, Parr, gravity, Rp2, vmr, vmrH2, vmrHe, mmw = calc_atmosphere( T0, alpha, logg, Mp, logvmr ) vmr_profile_h2o, vmr_profile_co, vmr_profile_ch4 = constant_vmr_profile(vmr) # cross sections for molecules xsm_h2o = opa_photo_h2o.xsmatrix(Tarr, Parr) xsm_co = opa_photo_co.xsmatrix(Tarr, Parr) xsm_ch4 = opa_photo_ch4.xsmatrix(Tarr, Parr) # sum of the opacity for molecules dtaum = art.opacity_profile_xs(xsm_h2o, vmr_profile_h2o, mmw, gravity) dtaum = dtaum + art.opacity_profile_xs(xsm_co, vmr_profile_co, mmw, gravity) dtaum = dtaum + art.opacity_profile_xs(xsm_ch4, vmr_profile_ch4, mmw, gravity) # CIAs dtauH2H2 = art.opacity_profile_cia( opa_photo_cia_H2H2.logacia_matrix(Tarr), Tarr, vmrH2, vmrH2, mmw, gravity ) dtauH2He = art.opacity_profile_cia( opa_photo_cia_H2He.logacia_matrix(Tarr), Tarr, vmrH2, vmrHe, mmw, gravity ) # sum of the opacity dtau_photo = dtaum + dtauH2H2 + dtauH2He F0 = art.run(dtau_photo, Tarr, nu_grid=nu_grid_photo) Fobs = F0 * (Rp2 / distance**2) * (RJ / pc) ** 2 mag = sop_photo.apparent_magnitude(Fobs) return mag .. GENERATED FROM PYTHON SOURCE LINES 319-324 checks the forward models ---------------------------- We check the forward models, `fspec` and `fphoto`, using an arbitrary parameters. .. GENERATED FROM PYTHON SOURCE LINES 324-376 .. code-block:: Python # just examples Mp = 33.2 RV = 28.1 T0 = 1293.0 alpha = 0.098 logvmr_sample = [-3.06, -2.83, -7.65] logg = 4.9 vsini = 16.2 spec_model = fspec( T0=T0, alpha=alpha, logg=logg, Mp=Mp, logvmr=logvmr_sample, u1=0.0, u2=0.0, RV=RV, vsini=vsini, ) print(spec_model) mu = spec_model err_all = f_obs_err a = jnp.dot(mu / err_all, f_obs / err_all) / jnp.dot(mu / err_all, mu / err_all) Kmag_model = fphoto( T0=T0, alpha=alpha, logg=logg, Mp=Mp, logvmr=logvmr_sample, ) print(Kmag_model) # fig = plt.figure(figsize=(12, 4)) fig = plt.figure(figsize=(20, 4)) plt.plot(nu_grid_obs, f_obs, ".", c="k", label="data", alpha=0.3) plt.plot( nu_grid_obs, a * spec_model, label="check", lw=2, alpha=0.7, ) plt.ylim(0.4, 1.3) plt.title("Kmag (model)=" + str(Kmag_model) + ", Kmag (obs)=" + str(Kmag)) plt.legend() plt.savefig("spec.png", bbox_inches="tight") .. GENERATED FROM PYTHON SOURCE LINES 377-385 Bayesian analysis --------------------- We define the probabilistic model using NumPyro. `fspec` is used as the mean of the Gaussian. Since the observed spectrum is a relative spectrum, the output of `fspec` must be normalized. Following Kawashima et al. (2025), we select a normalization constant that minimizes the residuals (strictly speaking, chi-square) between the model and the observed spectrum. .. GENERATED FROM PYTHON SOURCE LINES 385-427 .. code-block:: Python from jax import random import numpyro import numpyro.distributions as dist from numpyro.infer import MCMC, NUTS from numpyro.infer import Predictive from numpyro.diagnostics import hpdi mols_unique = ["H2O", "CO", "CH4"] def model_c(nu_grid_obs, y1, y1err, y2, y2err): logg = numpyro.sample("logg", dist.Uniform(4.0, 6.0)) Mp = numpyro.sample("Mp", dist.Normal(33.5, 0.3)) RV = numpyro.sample("RV", dist.Uniform(26, 30)) T0 = numpyro.sample("T0", dist.Uniform(1000.0, 2500.0)) alpha = numpyro.sample("alpha", dist.Uniform(0.05, 0.15)) vsini = numpyro.sample("vsini", dist.Uniform(10.0, 20.0)) logvmr = [] for i in range(len(mols_unique)): logvmr.append(numpyro.sample("log" + mols_unique[i], dist.Uniform(-10.0, 0.0))) u1 = 0.0 u2 = 0.0 sigma = numpyro.sample("sigma", dist.Exponential(10.0)) sig = jnp.ones_like(nu_grid_obs) * sigma err_all = jnp.sqrt(y1err**2.0 + sig**2.0) mu = fspec(T0, alpha, logg, Mp, logvmr, u1, u2, RV, vsini) # Normalizes the model spectrum to the data # The following equation is the solution of d/da |(a mu - fobs)/err_all|^2 = 0 for a a = jnp.dot(mu / err_all, f_obs[0] / err_all) / jnp.dot(mu / err_all, mu / err_all) numpyro.deterministic("a", a) mu = a * mu # photometry Kmag = fphoto(T0, alpha, logg, Mp, logvmr) numpyro.sample("y1", dist.Normal(mu, err_all), obs=y1) numpyro.sample("y2", dist.Normal(Kmag, y2err), obs=y2) .. GENERATED FROM PYTHON SOURCE LINES 428-438 MCMC sampling ---------------- We perform MCMC sampling using NUTS. It will take tens of hours or more. One way to check whether NUTS is running properly is to log the process and periodically monitor it using tools like `ezlog` (https://github.com/HajimeKawahara/ezlog). If the number of steps per chain becomes excessively large, some adjustments may be necessary. Based on experience, the number of steps per chain tends to increase during the warm-up phase, then decrease and stabilize at a certain level. .. GENERATED FROM PYTHON SOURCE LINES 438-455 .. code-block:: Python rng_key = random.PRNGKey(0) rng_key, rng_key_ = random.split(rng_key) # num_warmup, num_samples = 25, 100 num_warmup, num_samples = 500, 500 kernel = NUTS(model_c, forward_mode_differentiation=True) mcmc = MCMC(kernel, num_warmup=num_warmup, num_samples=num_samples) mcmc.run( rng_key_, nu_grid_obs=nu_grid_obs, y1=f_obs, y1err=f_obs_err, y2=Kmag, y2err=Kmag_err, ) mcmc.print_summary() .. GENERATED FROM PYTHON SOURCE LINES 456-461 saving and plotting ---------------------- We save the MCMC results and the samples and plot the trace. .. GENERATED FROM PYTHON SOURCE LINES 461-499 .. code-block:: Python import arviz as az az.plot_trace(mcmc, backend_kwargs={"constrained_layout": True}) plt.savefig("./output/trace.pdf", bbox_inches="tight") import pickle with open("./output_bn/mcmc.pickle", mode="wb") as f: pickle.dump(mcmc, f) with open("./output_bn/samples.pickle", mode="wb") as f: pickle.dump(mcmc.get_samples(), f) with open("./output_bn/samples.pickle", mode="rb") as f: samples = pickle.load(f) from numpyro.diagnostics import hpdi pred = Predictive(model_c, samples, return_sites=["y1", "y2"]) predictions = pred( rng_key_, nu_grid_obs=nu_grid_obs, y1=None, y1err=f_obs_err, y2=None, y2err=Kmag_err, ) with open("./output_bn/pred.pickle", mode="wb") as f: pickle.dump(predictions, f) median_mu1 = jnp.median(predictions["y1"], axis=0) hpdi_mu1 = hpdi(predictions["y1"], 0.95) np.savez("./output_bn/all.npz", [median_mu1, hpdi_mu1[0], hpdi_mu1[1]]) median_mu2 = jnp.median(predictions["y2"], axis=0) hpdi_mu2 = hpdi(predictions["y2"], 0.95) np.savez("./output_bn/Kmag.npz", [median_mu2, hpdi_mu2[0], hpdi_mu2[1]]) .. _sphx_glr_download_examples_spec_with_photo.py: .. only:: html .. container:: sphx-glr-footer sphx-glr-footer-example .. container:: sphx-glr-download sphx-glr-download-jupyter :download:`Download Jupyter notebook: spec_with_photo.ipynb ` .. container:: sphx-glr-download sphx-glr-download-python :download:`Download Python source code: spec_with_photo.py ` .. container:: sphx-glr-download sphx-glr-download-zip :download:`Download zipped: spec_with_photo.zip ` .. only:: html .. rst-class:: sphx-glr-signature `Gallery generated by Sphinx-Gallery `_