debian-skyfield/skyfield/sgp4lib.py

325 lines
12 KiB
Python

# -*- coding: utf-8 -*-
"""An interface between Skyfield and the Python ``sgp4`` library."""
from numpy import (
array, concatenate, identity, multiply, ones_like, repeat, zeros_like
)
from sgp4.api import SGP4_ERRORS, Satrec
from .constants import AU_KM, DAY_S, T0, tau
from .functions import mxv, rot_x, rot_y, rot_z
from .io_timescale import _build_builtin_timescale
from .positionlib import ITRF_to_GCRS2
from .searchlib import _find_discrete, find_maxima
from .timelib import calendar_date
from .vectorlib import VectorFunction
_identity = identity(3)
class EarthSatellite(VectorFunction):
"""An Earth satellite loaded from a TLE file and propagated with SGP4.
An earth satellite object is a Skyfield vector function, so you can
either call its ``at()`` method to generate its position in the sky
or else use addition and subtraction to combine it with other
vectors.
Satellite parameters are generally only accurate for a week or two
around the *epoch* of the parameters, the date for which they were
generated, which is available as an attribute:
``epoch``
A Skyfield :class:`~skyfield.timelib.Time` giving the exact
epoch moment for these satellite orbit parameters.
When building a satellite, use the arguments ``line1`` and ``line2``
to provide the two data lines from a TLE file as separate strings.
Optional ``name`` lets you give a name to the satellite, accessible
later through the ``name`` attribute. ``ts`` is a
:class:`~skyfield.timelib.Timescale` object, used to generate the
``epoch`` value; if it is not provided, the satellite will use a
built in ``Timescale`` object.
If you are interested in the catalog entry details, the SGP4 model
parameters for a particular satellite can be accessed through its
``model`` attribute:
``model.satnum``
The unique satellite NORAD catalog number given in the TLE file.
``model.epochyr``
Full four-digit year of this element set's epoch moment.
``model.epochdays``
Fractional days into the year of the epoch moment.
``model.jdsatepoch``
Julian date of the epoch (computed from ``epochyr`` and ``epochdays``).
``model.ndot``
First time derivative of the mean motion (ignored by SGP4).
``model.nddot``
Second time derivative of the mean motion (ignored by SGP4).
``model.bstar``
Ballistic drag coefficient B* in inverse earth radii.
``model.inclo``
Inclination in radians.
``model.nodeo``
Right ascension of ascending node in radians.
``model.ecco``
Eccentricity.
``model.argpo``
Argument of perigee in radians.
``model.mo``
Mean anomaly in radians.
``model.no_kozai``
Mean motion in radians per minute.
"""
center = 399
ts = None # see __init__()
def __init__(self, line1, line2, name=None, ts=None):
if ts is None:
ts = self.ts
if ts is None:
ts = EarthSatellite.ts = _build_builtin_timescale()
self.name = None if name is None else name.strip()
satrec = Satrec.twoline2rv(line1, line2)
self.model = satrec
two_digit_year = satrec.epochyr
if two_digit_year < 57:
year = two_digit_year + 2000
else:
year = two_digit_year + 1900
self.epoch = ts.utc(year, 1, satrec.epochdays)
self._setup(satrec)
def _setup(self, satrec):
# If only I had not made __init__() specific to TLE lines, but
# had put them in an alternate construtor instead, this would
# simply have lived in __init__(). Alas! I was so young then.
self.target = -100000 - satrec.satnum
@classmethod
def from_satrec(cls, satrec, ts):
"""Build an EarthSatellite from a raw sgp4 Satrec object.
This lets you provide raw numeric orbital elements instead of
the text of a TLE set. See :ref:`from-satrec` for detais.
"""
self = cls.__new__(cls)
self.model = satrec
self.name = None
# TODO: once sgp4 starts filling in epochyr and epochdays in
# sgp4init(), the separate epoch code here and in __init__() can
# be unified to always use epochyr and epochdays.
year, month, day = calendar_date(satrec.jdsatepoch)
self.epoch = ts.utc(year, month, day + satrec.jdsatepochF)
self._setup(satrec)
return self
def __str__(self):
sat = self.model
return 'EarthSatellite{0} number={1!r} epoch={2}'.format(
' ' + repr(self.name) if self.name else '',
sat.satnum,
self.epoch.utc_iso(),
)
def __repr__(self):
return '<{0}>'.format(self)
def _position_and_velocity_TEME_km(self, t):
"""Return the raw true equator mean equinox (TEME) vectors from SGP4.
Returns a tuple of NumPy arrays ``([x y z], [xdot ydot zdot])``
expressed in kilometers and kilometers per second. Note that we
assume the TLE epoch to be a UTC date, per AIAA 2006-6753.
"""
sat = self.model
jd = t._utc_float()
if getattr(jd, 'shape', None):
e, r, v = sat.sgp4_array(jd, zeros_like(jd))
messages = [SGP4_ERRORS[error] if error else None for error in e]
return r.T, v.T, messages
else:
error, position, velocity = sat.sgp4(jd, 0.0)
message = SGP4_ERRORS[error] if error else None
return array(position), array(velocity), message
def ITRF_position_velocity_error(self, t):
"""Return the ITRF position, velocity, and error at time `t`.
The position is an x,y,z vector measured in au, the velocity is
an x,y,z vector measured in au/day, and the error is a vector of
possible error messages for the time or vector of times `t`.
"""
rTEME, vTEME, error = self._position_and_velocity_TEME_km(t)
rTEME /= AU_KM
vTEME /= AU_KM
vTEME *= DAY_S
rITRF, vITRF = TEME_to_ITRF(t.ut1, rTEME, vTEME)
return rITRF, vITRF, error
def _at(self, t):
"""Compute this satellite's GCRS position and velocity at time `t`."""
rITRF, vITRF, error = self.ITRF_position_velocity_error(t)
rGCRS, vGCRS = ITRF_to_GCRS2(t, rITRF, vITRF)
return rGCRS, vGCRS, rGCRS, error
def find_events(self, topos, t0, t1, altitude_degrees=0.0):
"""Return the times at which the satellite rises, culminates, and sets.
Searches between ``t0`` and ``t1``, which should each be a
Skyfield :class:`~skyfield.timelib.Time` object, for passes of
this satellite above the location ``topos`` that reach at least
``altitude_degrees`` above the horizon.
Returns a tuple ``(t, events)`` whose first element is a
:class:`~skyfield.timelib.Time` array and whose second element
is an array of events:
* 0 — Satellite rose above ``altitude_degrees``.
* 1 — Satellite culminated and started to descend again.
* 2 — Satellite fell below ``altitude_degrees``.
Note that multiple culminations in a row are possible when,
without setting, the satellite reaches a second peak altitude
after descending partway down the sky from the first one.
"""
# First, we find the moments of maximum altitude over the time
# period. Some of these maxima will be negative, meaning the
# satellite failed to crest the horizon.
ts = t0.ts
at = (self - topos).at
half_second = 0.5 / DAY_S
orbits_per_minute = self.model.no_kozai / tau
orbits_per_day = 24 * 60 * orbits_per_minute
# Note the protection against zero orbits_per_day. I have not
# yet learned with which satellite caused a user to run into a
# ZeroDivisionError here, and without a concrete example I have
# little sense of whether 1.0 is a good fallback value or not.
rough_period = 1.0 / orbits_per_day if orbits_per_day else 1.0
# Long-period satellites might rise each day not because of
# their own motion, but because the Earth rotates under them, so
# check position at least each quarter-day. We might need to
# tighten this even further if experience someday shows it
# missing a pass of a particular satellite.
if rough_period > 0.25:
rough_period = 0.25
def cheat(t):
"""Avoid computing expensive values that cancel out anyway."""
t.gast = t.tt * 0.0
t.M = t.MT = _identity
def altitude_at(t):
cheat(t)
return at(t).altaz()[0].degrees
altitude_at.rough_period = rough_period
tmax, altitude = find_maxima(t0, t1, altitude_at, half_second, 12)
if not tmax:
return tmax, ones_like(tmax)
# Next, filter out the maxima that are not high enough.
keepers = altitude >= altitude_degrees
jdmax = tmax.tt[keepers]
ones = ones_like(jdmax, 'uint8')
# Finally, find the rising and setting that bracket each maximum
# altitude. We guess that the satellite will be back below the
# horizon in between each pair of adjancent maxima.
def below_horizon_at(t):
cheat(t)
return at(t).altaz()[0].degrees < altitude_degrees
# The `jdo` array are the times of maxima, with their averages
# in between them. The start and end times are thrown in too,
# in case a rising or setting is lingering out between a maxima
# and the ends of our range. Could this perhaps still miss a
# stubborn rising or setting near the ends?
doublets = repeat(concatenate(((t0.tt,), jdmax, (t1.tt,))), 2)
jdo = (doublets[:-1] + doublets[1:]) / 2.0
trs, rs = _find_discrete(t0.ts, jdo, below_horizon_at, half_second, 8)
jd = concatenate((jdmax, trs.tt))
v = concatenate((ones, rs * 2))
i = jd.argsort()
return ts.tt_jd(jd[i]), v[i]
def theta_GMST1982(jd_ut1):
"""Return the angle of Greenwich Mean Standard Time 1982 given the JD.
This angle defines the difference between the idiosyncratic True
Equator Mean Equinox (TEME) frame of reference used by SGP4 and the
more standard Pseudo Earth Fixed (PEF) frame of reference.
From AIAA 2006-6753 Appendix C.
"""
t = (jd_ut1 - T0) / 36525.0
g = 67310.54841 + (8640184.812866 + (0.093104 + (-6.2e-6) * t) * t) * t
dg = 8640184.812866 + (0.093104 * 2.0 + (-6.2e-6 * 3.0) * t) * t
theta = (jd_ut1 % 1.0 + g / DAY_S % 1.0) % 1.0 * tau
theta_dot = (1.0 + dg / (DAY_S * 36525.0)) * tau
return theta, theta_dot
_zero_zero_minus_one = array((0.0, 0.0, -1.0))
_cross120 = array((1,2,0))
_cross201 = array((2,0,1))
def _cross(a, b):
# Nearly 4x speedup over numpy cross(). TODO: Maybe move to .functions?
return a[_cross120] * b[_cross201] - a[_cross201] * b[_cross120]
def TEME_to_ITRF(jd_ut1, rTEME, vTEME, xp=0.0, yp=0.0):
"""Convert TEME position and velocity into standard ITRS coordinates.
This converts a position and velocity vector in the idiosyncratic
True Equator Mean Equinox (TEME) frame of reference used by the SGP4
theory into vectors into the more standard ITRS frame of reference.
The velocity should be provided in units per day, not per second.
From AIAA 2006-6753 Appendix C.
"""
# TODO: are xp and yp the values from the IERS? Or from general
# nutation theory?
theta, theta_dot = theta_GMST1982(jd_ut1)
angular_velocity = multiply.outer(_zero_zero_minus_one, theta_dot)
R = rot_z(-theta)
if len(rTEME.shape) == 1:
rPEF = (R).dot(rTEME)
vPEF = (R).dot(vTEME) + _cross(angular_velocity, rPEF)
else:
rPEF = mxv(R, rTEME)
vPEF = mxv(R, vTEME) + _cross(angular_velocity, rPEF)
if xp == 0.0 and yp == 0.0:
rITRF = rPEF
vITRF = vPEF
else:
W = (rot_x(yp)).dot(rot_y(xp))
rITRF = (W).dot(rPEF)
vITRF = (W).dot(vPEF)
return rITRF, vITRF