Introduction to the IAU2000/2006 precession-nutation theory

Introduction to the IAU2000/2006 precession-nutation theory#

Added in version 7.3.0.

heyoka.py provides an implementation of the IAU2000/2006 precession-nutation theory. Precession and nutation are periodic variations in the orientation of the Earth’s rotational axis caused by the gravitational forces of the Sun and the Moon, and to a lesser extent other bodies.

IAU2000/2006 consists of a set of analytical formulae representing the three angles \(X\), \(Y\) and \(s\) as a function of time.

\(X\) and \(Y\) are the celestial intermediate pole (CIP) coordinates in the International Celestial Reference System (ICRS). They describe the position of the CIP relative to the Geocentric Celestial Reference Frame (GCRF), which is aligned with the ICRS.

\(s\) is the Celestial Intermediate Origin (CIO) locator. It represents an additional rotation in the transformation between celestial and terrestrial reference frames.

For more information about these quantities, please consult standard references such as Vallado’s “Fundamentals of Astrodynamics” (Chapter 3) and Chapter 5 of the IERS conventions.

API overview#

The heyoka.model.iau2006() function returns analytical expressions for the \(X\), \(Y\) and \(s\) angles as functions of the input time. The angles are returned in radians.

Precision and truncation threshold#

The IAU2000/2006 theory is formulated as a set of large trigonometric series whose numerical coefficients decay exponentially. By dropping the smaller terms from the series, the precision of the theory degrades but computational efficiency improves. By default, iau2006() drops all trigonometric terms whose coefficients have a magnitude less than 1 microarcsecond. The truncation threshold can be changed via the thresh keyword argument (measured in arcseconds). See below for an analysis of the accuracy of the theory at several truncation levels.

Time coordinate#

Although technically IAU2000/2006 uses barycentric dynamical time (TDB), it is common praxis to use terrestrial time (TT) instead - the two timescales are sufficiently close so as not to introduce any noticeable error in the solutions. Time is measured in Julian centuries from J2000 (JD2451545.0).

By default, iau2006() uses heyoka.time to represent the time variable in the IAU2000/2006 formulae. This means that, by default, when the IAU2000/2006 solution is used in an ODE system, time is assumed to be measured in hundreds of Julian years and \(t = 0\) corresponds to the Julian date \(2451545.0\).

It is possible to change the expression used to represent the time variable in the IAU2000/2006 solution via the time_expr keyword argument. This allows to rescale and change the origin of time in the IAU2000/2006 formulae.

For instance, if we want time to be measured in Julian days (rather than centuries) since J2000, we can write:

import heyoka as hy

hy.model.iau2006(time_expr=hy.time / 36525.0);

In a similar fashion, we can change the origin of the time coordinate by adding/subtracting an offset to heyoka.time. For instance, if we want to be able to use directly the Julian date as time variable (rather than the number of Julian days since J2000), we can write:

hy.model.iau2006(time_expr=(hy.time - 2451545.0) / 36525.0);

Note that the time expression passed to the iau2006() function does not need to be based on heyoka.time - in fact any expression can be used as a time coordinate in the IAU2000/2006 formulae.

Truncation threshold and accuracy#

In this section, we will provide an assessment of the effect of truncation threshold on the accuracy of the theory.

We begin by introducing a compiled function for the evaluation of the full theory (i.e., without truncation):

# Create a heyoka variable for representing time.
tm = hy.make_vars("tm")

# Construct a compiled function for the full theory.
# We use the Julian date as a time coordinate.
cf_full = hy.cfunc(
    hy.model.iau2006(time_expr=(tm - 2451545.0) / 36525.0, thresh=0.0),
    [tm],
    compact_mode=True,
)

heyoka.py’s implementation of the full theory has been validated against the standard ERFA/SOFA routines.

Next, we define a timespan of Julian dates at which we will be evaluating the theory:

import numpy as np

# We will perform 1000 evaluations of the theory
# in the 10 years following J2000.
dates = np.linspace(2451545.0, 2451545.0 + 365.25 * 10, 1000)

We can now proceed to evaluate the full theory over the timespan:

# Evaluate the full theory.
full_eval = cf_full([dates])

Finally, we will be evaluating the theory at several nonzero truncation levels and compare the results with the full theory:

%matplotlib inline
from matplotlib.pylab import plt
import numpy as np

# Setup the plot.
fig = plt.figure(figsize=(9, 9))

# Threshold levels at which we will be computing
# the IAU2000/2006 solution.
thr_values = [1e-5, 1e-6, 1e-7]

for idx, thr in enumerate(thr_values):
    # Build the IAU2000/2006 theory, using the Julian
    # date as a time coordinate and setting the threshold
    # level.
    X, Y, s = hy.model.iau2006(time_expr=(tm - 2451545.0) / 36525.0, thresh=thr)

    # Compile the function for the evaluation of
    # the truncated theory.
    cf = hy.cfunc([X, Y, s], [tm], compact_mode=True)

    # Run the evaluation.
    X_Y_s_values = cf([dates])

    # Compute the difference wrt the full theory in
    # arcseconds.
    diff = np.abs(full_eval - X_Y_s_values) * 180 * 3600 / np.pi

    # Plot.
    plt.subplot(len(thr_values), 1, idx + 1)
    plt.yscale("log")
    plt.plot(dates, diff[0], "r", label="X")
    plt.plot(dates, diff[1], "k", label="Y")
    plt.plot(dates, diff[2], "b", label="s")

    ax = plt.gca()
    ax.set_title(f"Threshold={thr}")

    if idx == 2:
        plt.xlabel("Time (JD)")
        plt.legend()
    else:
        plt.xticks([])
    plt.ylabel("Error (arcseconds)")

plt.tight_layout();
../_images/bc9a2d23bd31f03c606d9e6d3d429e47b9d407b1e3f83dd06a54e76f40755eae.png

We can clearly see the effects of the truncation threshold on the accuracy of the theory with respect to the full solution. For reference, an error of \(10^{-3}\) arcseconds translates on the Earth’s surface to a positional error of circa 3 centimeters.