Syzygy.jl
is a high-performance few-body simulator written in Julia. This code is mainly aimed at simulating and visualizing the dynamics of hierarchical multistar systems, but it also supports non-hierarchical few-body systems. The package uses the DifferentialEquations.jl ecosystem to solve the governing differential equations. This makes the package highly performant and flexible, allowing for, e.g., both fixed and adaptive timestepping with adjustable error tolerances, and the flexibility of callbacks for code injection.
Caution
This package is still being developed, so certain features may not work as expected. If you want to use this package, please ensure that you perform tests to ensure that your results are to be expected.
Syzygy.jl
is not a registered package, which means it can be installed using the link to this Github repo. To install it in the REPL, enter the Pkg mode with ]
and just type
julia> ] # enter Pkg mode
pkg> add https://github.com/casparwb/Syzygy.jl
Alternatively, from your terminal run
> julia -e 'import Pkg; Pkg.add(url="https://github.com/casparwb/Syzygy.jl")'
A simulation in Syzygy.jl
begins by setting up the system you want to simulate. This is done by the multibodysystem
function, which takes in the structural arguments of the bodies in the system - masses, radii, stellar types, luminosities, etc... -, and the orbital parameters of the binaries - semi-major axes, eccentricities, etc.. . The masses are set as the first positional argument, while all other parameters are set using keyword arguments. Syzygy.jl
uses units by utilizing Unitful.jl and UnitfulAstro.jl, which of which are re-exported upon loading Syzygy.jl
. Arguments that are not unitless need to be defined with a unit when initializing the system. The system is set up following Hamers & Portegies Zwart 2016, with labeling being done as shown in this figure (Evans 1968):

Important
The code for setting up the hierarchy (converting the orbital elements into state vectors) is either heavily inspired by, or directly taken from NbodyGradient.jl. All credits go to the authors.
binary = multibodysystem([1.0, 1.0]u"Msun", a=1.0u"AU", e=0.4) # set up a binary system with two 1 solar-mass stars, in an orbit with 1 semi-major axis of 1 AU and an eccentricity of 0.4
triple = multibodysystem([2.0, 1.0, 3.0]u"Msun", a=[0.1, 0.5]u"Rsun", e=[0.1, 0.4], i=[π/2, 0.0]u"rad") # hierarchical triple
quadruple = multibodysystem([1.0, 1.0, 1.0, 1.0]u"Msun", a=[0.1, 0.5, 10.0]u"Rsun", e=[0.1, 0.4, 0.2], i=[90.0, 45.0, 0.0]u"degree", hierarchy=[4, 2, 1]) # 2+2 quadruple
# set up a binary black hole system
bh1_mass = 9.62u"Msun"
bh2_mass = 8.4u"Msun"
bh1_radius = 2*GRAVCONST*bh1_mass/c² # the gravitational constant G can be accessed using GRAVCONST in the REPL.
bh2_radius = 2*GRAVCONST*bh2_mass/c²
binary_blackholes = multibodysystem([bh1_mass, bh2_mass], a=15.3u"Rsun", R=[bh1_radius, bh2_radius], stellar_type=[14, 14])
If you wanted to set up the system in the above figure, a quintuple system, you would do
masses = ones(5)u"Msun"
r1 = 1.0u"Rsun"
r2 = 5.0u"Rsun"
r3 = 1.0u"Rsun"
r4 = 10.0u"Rsun"
hierarchy = [5, 1, 2, 1] # 1 binary on level 2, 2 on 1, and 1 on 2
quintuple = multibodysystem(masses, a=[r1, r2, r3, r4], hierarchy=hierarchy)
The components of the system can be accessed via system.binaries[index]
, with index
being an integer from 1 to number of binaries, and system.particles[index]
, for the individual particles. A binary or particle are instances of the Binary
or Particle
type, both of which have a number of fields containing information about their state and internal structure.
A system can also be initialized using just masses, positions, and velocities (in addition to structural arguments like radius etc.). To do this, call multibodysystem
with positional arguments masses, positions, velocities.
masses = [2.0, 1.0]u"Msun"
r1 = [-1.0, 0.0, 0.0]u"Rsun"
r2 = [1.0, 0.0, 0.0]u"Rsun"
v1 = [0.0, -25.0, 0.0]u"km/s"
v2 = [0.0, 25.0, 0.0]u"km/s"
twobody = multibodysystem(masses, [r1, r2], [v1, v2], R=[1.5, 0.5]u"Rsun")
Note
Syzygy.jl
uses units when initializing a system and when postprocessing a simulation results. The units are discarded before the actual simulation begins in order to ensure performance. The initial conditions and parameters that are sent to the ODE solver are always the preferred units of the respective dimensions. The default unit system is solar mass (preferunits
function from Unitful.jl
BEFORE importing Syzygy.jl
.
Once the system has been initialized, you can set up the parameters of the n-body simulation using the simulation
function, which takes in the system as the first and only positional argument, followed by keyword arguments for all the other properties. simulation
accepts almost any argument supported by the common solver options of the solve
function from DifferentialEquations.jl. See the Docs for a full overview. Note that solvers are not re-exported, thus they can be specified with the Syzygy
prefix, like Syzygy.DPRKN6()
.
t_sim
: total simulation time, which can either be aNumber
, in which case the simulation time will bet_sim
multiplied by the period of the outermost binary, or it can be aQuantity
, i.e., a number with a (time) unit, in which case the simulation time is simplyt_sim
.npoints
: number of snapshots to save. With this parameter, a snapshot will be taken every(t_end - t_start)/npoints
, and will therefore be uniformly distributed in time. You can also use thesaveat
argument fromDiffEq
, which can take in an array of specific times, or other types.alg
: the solver to use. Default is the 8-th order Runge-Kyutta-Nyström method as described in the DiffEq 2nd order ODE solver docs. This is an adaptive solver, thus the only solver arguments are the absolute and relative error tolerances, which can be set usingabs_tol
andrel_tol
respectively. By default they are both set to1e-10
.dt
: time step. Only used if the solver is not adaptive, such asVerletLeapfrog
,McAte5
,Yoshida6
, etc.. Can either be aNumber
or aQuantity
, where aNumber
would specify a timestep equal todt
multiplied by the innermost orbital period.showprogress
: whether to display a progress tracker, showing the current absolute system time and its percentage oft_sim
. Will slow down the simulation.verbose
: whether to output information about the setup and the final outcome, including total runtime and energy loss.callbacks
: see section on callbacks.potentials
: see section on potentials.
sim = simulation(triple, t_sim=10, npoints=10_000) # simulate for 10 outer orbits, and save 10 000 snapshots uniformly distributed
sim = simulation(triple, t_sim=10u"kyr", save_everystep=false) # simulate for 10 000 years, and only save the initial and final state of the system
sim = simulation(triple, t_sim=1, alg=Syzygy.McAte5(), dt=1e-5) # use a timestep of 1e-5 * P_in
The package contains several pre-defined callbacks that can be specified when setting up the simulation. These mostly define stopping conditions, but can also be e.g, flags or events. Callbacks can be specified using the callbacks
keyword argument when calling simulation
, and be given as a Vector{<:AbstractSyzygyCallback}
. There are number of pre-defined callbacks, but you can also define your own. (see Advanced usage for more details). The most important pre-defined callbacks are
CollisionCB
: a stopping condition that terminates the integration when two objects gets close enough. The exact condition for specifying a collision depends on the stellar types of the objects. If the two objects are both stars, then the code simply checks if the radii overlap. If the objects are stellar remnants (black holes, neutron stars, white dwarfs), then the code checks if the distance is smaller than some multiple of their mutual gravitational radius. By default, this multiple value is set to 1000. If the two objects contain one star and one remnant, then the code checks if the star is within the tidal disruption radius of the remnant object.EscapeCB
: a stopping condition for checking whether an object has been ejected from the system. The callback has three checks that need to be passed in order for the condition to be invoked. See Standish & Myles 1971 for more details. Currently this callback is only possible to use for a triple system.
Certain callbacks accepts argument when instantiating them. This can include how often the solver should check for the condition (for every n-th step). This is implemented as the field check_every
.
[!INFO] To get the full list of callbacks, you can call
Syzygy.callbacks()
, or you can do e.g.?CollisionCB
to see the documentation for each callback.
callbacks = [CollisionCB(), EscapeCB()]
sim = simulation(system, t_sim=1, callbacks=callbacks)
Syzygy.jl
has support for including/varying the potentials and corresponding acceleration functions in a simulation. In this package, you can specify the acceleration functions to include by setting the potentials
keyword argument when calling simulate
or simulation
. This argument accepts a Vector{MultiBodyPotential}
where MultiBodyPotential
is a supertype of all the potentials defined in the package, and for each of which there exists a unique acceleration function. The total acceleration of a particle at each time step is thus a sum of all the acceleration functions defined by the potential
vector.
Currently, there are a number of possible potentials:
PureGravitationalAcceleration
: Newtonian gravitational accelerationDynamicalTidalAcceleration
: energy dissipation from dynamical tides, following the prescription of Samsing et al. 2018.EquilbriumTidalAcceleration
: tidal drag force from equilibrium tides, as described by Hurley at al. 2002, using the model of Hut 1981.StaticEquilibriumTidalAcceleration
: same as above, but with the assumption that masses, radii, and other structural parameters of the particles do not change during the simulation. This is more efficient as it only calculates certain quantities once, and re-uses them throughout the simulation.
There are also potentials for various general relatistic effects via post-Newtonian expansion. See section on post-Newtonian physics
Each of these accepts a specific set of arguments related to the acceleration functions. To get an overview of these, enter the help
-mode in the REPL with ?
and type any of the above names.
Important
The numerical implementation of potentials and acceleration functions in this package is heavily inspired by or completely taken from NbodySimulator.jl. All credit go the authors.
grav_pot = PureGravitationalAcceleration()
tidal_pot = DynamicalTidalAcceleration(;kwargs...)
res = simulate(system, potential=[grav_pot, tidal_pot])
For systems where general relativistic effects can become important, Syzygy.jl
supports post-Newtonian acceleration potential up to order 2.5. It also has support for evolving the precession of the spins of compact objects, up to order 2. The full list of post-Newtonian potentials are given here:
PN1Potential
: post-Newtonian acceleration of order 1.PN2Potential
: post-Newtonian acceleration of order 2.PN2p5Potential
: post-Newtonian acceleration of order 2.5.PNPotential
: post-Newtonian acceleration of order 1 to 2.5. This is more efficient than including all of the above potentials individually.PN1SpinPrecessionPotential
: spin precession potential of order 1.PN1p5SpinPrecessionPotential
: spin precession potential of order 1.5.PN2SpinPrecessionPotential
: spin precession potential of order 2.SpinPrecessionPotential
: spin precession potential of order 1 to 2.
Once the simulation has been instantiated, it can be run using simulate(sim)
. You can also skip the previous step and directly call simulate
with the exact same arguments as for simulation
.
res = simulate(triple, t_sim=10, npoints=1000)
Once a simulation has finished, it can be analyzed and converted to a MultiBodySolution
type, which allows for easy access to the state vectors and other parameters. This is done using the analyse_simulation
function, which only takes in the resulting output from simulate
as its argument. In the MultiBodySolution
type, the state vectors are stored in 3-dimensional AxisArray
from AxisArrays.jl, with the axes being (dimension, particle, time).
sol = to_solution(res)
N = length(sol.t)
r1 = sol.r[particle=1] # a (3, N) array, alias for sol.r[:,1,:]
v1 = sol.v[particle=1]
Syzygy.jl
contains a few plot recipes for Plots.jl, which can be used to create quick and simple visualizations of the orbits. To use this, simply import Plots.jl
, and use the function orbitplot
, which takes in a MultiBodySolution
as its first argument, followed by keyword arguments:
dims
: which dimensions to plot. Should be aVector{Int}
, with any combination of(1, 2, 3)
(x, y, z).bodies
: which bodies (particles) to plot. Should be aVector{Int}
, with any combination of1:N_particles
.tspan
: the temporal selection to plot. Should be aTuple{Quantity, Quantity}
, with the start- and end-points.
using Plots
orbitplot(sol, bodies=[1, 2], dims=[1, 2]) # plot only particles 1 and 2 in the x-y plane
By default Syzygy.jl
uses Float64
datatypes for the variables of integration, which means that we can not set the error tolerances for the ODE solver lower than Syzygy.jl
supports higher and arbitracy arithmetic, which means you can essentially get as small error as you want aT the cost of speed. This is done through the (DoubleFloats.jl)[] and (ArbNumerics.jl)[] packages. You get the best performance using the Double64
type from the former package, which has twice the precision of Float64
s. If you need even higher precision, you can use the arbitrary precision floating point numbers from (ArbNumerics.jl)[https://github.com/JeffreySarnoff/ArbNumerics.jl]. However, using these will dramatically slow down the code. To use this functionality, simply specify the precision
keyword when calling either simulation
or simulate
. Fixed-bit types are specified with symbols, while arbitrary precision are specified with an integer, determining the number of bits to use.
binary = multibodysystem([2.0, 1.0]u"Msun", a=1.0u"AU")
result = simulate(binary, t_sim=1, save_everystep=false, precision=:Double64) # will use Double64 precision (106 bits)
result = simulate(binary, t_sim=1, save_everystep=false, precision=254) # will use ArbFloat type with 254 bit precision
Syzygy.jl
is designed to be highly composable and flexible, and not only includes additional acceleration functions for including other forces, but also allows the users to define their own potentials and callbacks.
To define you own callback, first define a struct which is a subtype of Syzygy.AbstractSyzygyCallback
. For example:
struct MyCallback <: Syzygy.AbstractSyzygyCallback end
You can then add a method to the Syzygy.get_callback
function, which has the signature get_callback(cb, system, retcodes, args)
, where cb
is an object with a type of your callback struct, system
is the system that is simulated, retcodes is a dictionary in which you can add return codes/flags, and args can be any dictionary with additional arguments. Inside this function, you set up your affect
and condition
functions, which determine when the callback is triggered, and what should happen when it does. It should then return a callback type from the SciMLBase package.
For example, if you want to add a callback that terminates the integration if it runs for a certain amount of time, you would do:
function Syzygy.get_callback(cb::MyCallback, system, retcodes, args)
max_cpu_time = args[:max_cpu_time]
t_start = time()
condition_cpu_time(u, t, integrator) = true # check every step
function max_cpu_time_callback!(integrator)
if (time() - t_start) >= max_cpu_time
retcode[:MaxCPUTime] = true
terminate!(integrator)
end
end
affect_cpu_time!(integrator) = max_cpu_time_callback!(integrator)
callback_cpu_time = DiscreteCallback(condition_cpu_time, affect_cpu_time!, save_positions=(false, false))
callback_cpu_time
end
To add a new potential and acceleration function, you first need to define the acceleration function itself, which should be an in-place function that calculates the pairwise acceleration for two particles nothing
.
Example:
function my_acceleration_function!(dvi, dvj, rs, vs, pair, params)
i, j = pair
mi, mj = params.M[i], params.M[j]
ai = acceleration(i, mj)
aj = acceleration(j, mi)
dvi .+= ai
dvj .+= aj
nothing
end
You then need to define a potential type, which must be a subtype of the Syzygy.MultiBodyPotential
supertype.
struct MyPotential{T1, T2} <: Syzygy.MultiBodyPotential end
Finally, you add a method to the Syzygy.get_accelerating_function
such that it returns a wrapper around your acceleration function. The wrapper has to have the following signature:
function Syzygy.get_accelerating_function(potential::MyPotential)
(dvi, dvj, rs, vs, pair, time, params)-> my_acceleration_function!(dvi, dvj, rs, vs, pair, params)
end
To use this potential in a simulation, create an instance of your new MyPotential
type, and include it in the potential
vector as a keyword argument when initializing the simulation.
mypotential = MyPotential()
res = simulate(system, potential=[PureGravitationalAcceleration(), MyPotential()])