Quick Start (@id quick-start)
This guide introduces the key concepts in Octofitter:
- Likelihood objects to hold your data
- Planetand- Systemmodels to specify variables, priors, and system architecture
- Sampling from the posterior using MCMC
- Plotting the results
- Saving the chain
For installation instructions, see Installation.
Example: Fit a Single Planet Orbit to Relative Astrometry
Load the required packages:
using Octofitter, Distributions, CairoMakie, PairPlotsCreate a PlanetRelAstromLikelihood object containing your observational data. In this case its the position of the planet relative to the star, but many other kinds of data are supported:
astrom_dat = Table(
    epoch = [50000, 50120, 50240],      # Dates in MJD
    ra = [-505.7, -502.5, -498.2],      # [mas] East positive
    dec = [-66.9, -37.4, -7.9],         # [mas] North positive
    σ_ra = [10.0, 10.0, 10.0],          # [mas] Uncertainties
    σ_dec = [10.0, 10.0, 10.0],         # [mas] Uncertainties
    cor = [0.0, 0.0, 0.0]               # RA/Dec correlations
)
astrom = PlanetRelAstromLikelihood(astrom_dat, name="GPI astrom") # must give a name for each group of observationsDefine a planet model with orbital elements and their prior distributions:
planet_b = Planet(
    name="b",
    basis=Visual{KepOrbit},
    likelihoods=[astrom],
    variables=@variables begin
        M ~ truncated(Normal(1.2, 0.1), lower=0.1)  # Total mass (solar masses) for this orbit
        a ~ Uniform(0, 100)        # Semi-major axis [AU]
        e ~ Uniform(0.0, 0.5)      # Eccentricity  
        i ~ Sine()                 # Inclination [rad]
        ω ~ UniformCircular()      # Argument of periastron [rad]
        Ω ~ UniformCircular()      # Longitude of ascending node [rad]
        θ ~ UniformCircular()      # Position angle at reference epoch [rad]
        # Epoch of periastron passage
        # We calculate it from the position angle above
        tp = θ_at_epoch_to_tperi(θ, 50000; M, e, a, i, ω, Ω)  
    end
)Define the system with its mass and distance - see System Construction for more options:
sys = System(
    name="HD1234",
    companions=[planet_b],
    likelihoods=[],
    variables=@variables begin
        plx ~ truncated(Normal(50.0, 0.02), lower=0.1)  # Parallax (mas)
    end
)That there are many different orbit parameterizations, each requiring different of parameters names. The KepOrbit is a full 3D keplerian orbit with Campbell parameters. Visual means that we have a defined parallax distance plx that can map separations in AU to arcseconds.
Compile the model into efficient sampling code:
model = Octofitter.LogDensityModel(sys)Initialize the starting points for the chains. You can optionally provide starting values directly for certain variables (UniformCircular priors are a special case, see docs for details).
init_chain = initialize!(model, (;
    plx = 50.001,
    planets = (;
        b=(;
            M = 1.18,
            a = 10.0,
            e = 0.01,
        )
    )
)) Visualize the starting point. You can use this plot to make absolutely sure your data is entered in correctly:
octoplot(model, init_chain)Sample from the posterior using Hamiltonian Monte Carlo (see Samplers for other options):
chain = octofit(model, iterations=1000)Visualize the results with orbit plots and a corner plot:
octoplot(model, chain)     # Plot orbits and dataoctocorner(model, chain)   # Corner plot of posteriorSave the results to a FITS file (see Loading and Saving Data for other formats):
Octofitter.savechain("output.fits", chain)Working with Dates
These functions may help you convert dates to and from Modified Julian Days using these helper functions:
mjd("2020-01-01")     # Date string to MJD
years2mjd(2020.0)     # Decimal year to MJD
mjd2date(50000)       # MJD to dateNext Steps
See the Tutorials section for complete examples.