1use crate::dynamics::DynamicsError;
20pub use crate::dynamics::{Dynamics, NyxError};
21use crate::errors::StateError;
22use crate::io::{ConfigError, InputOutputError};
23use crate::linalg::OVector;
24use crate::md::prelude::SpacecraftDynamics;
25use crate::md::trajectory::TrajError;
26use crate::propagators::PropagationError;
27use crate::time::Epoch;
28pub use crate::{State, TimeTagged};
29use anise::almanac::planetary::PlanetaryDataError;
30use anise::errors::AlmanacError;
31use hifitime::Duration;
32use snafu::prelude::Snafu;
33use std::sync::Arc;
34
35pub mod kalman;
36
37mod ground_station;
39pub use ground_station::GroundStation;
40
41pub mod estimate;
43
44pub mod noise;
46
47pub mod msr;
49
50pub mod simulator;
52
53pub mod process;
55
56pub use simulator::TrackingDevice;
57
58pub mod snc;
60
61pub mod blse;
63
64#[cfg(feature = "premium")]
66pub mod interlink;
67
68#[cfg(feature = "premium")]
70pub mod groundpnt;
71
72#[cfg(feature = "premium")]
74pub mod position;
75
76pub type SpacecraftKalmanOD = self::process::KalmanODProcess<
78 SpacecraftDynamics,
79 nalgebra::Const<2>,
80 nalgebra::Const<3>,
81 GroundStation,
82>;
83
84pub type SpacecraftKalmanScalarOD = self::process::KalmanODProcess<
86 SpacecraftDynamics,
87 nalgebra::Const<1>,
88 nalgebra::Const<3>,
89 GroundStation,
90>;
91
92#[cfg(feature = "premium")]
94pub type InterlinkKalmanOD = self::process::KalmanODProcess<
95 SpacecraftDynamics,
96 nalgebra::Const<2>,
97 nalgebra::Const<3>,
98 interlink::InterlinkTxSpacecraft,
99>;
100
101#[allow(unused_imports)]
102pub mod prelude {
103 pub use super::estimate::*;
104 pub use super::ground_station::*;
105 pub use super::kalman::KalmanVariant;
106 pub use super::kalman::*;
107 pub use super::msr::*;
108 pub use super::noise::{GaussMarkov, StochasticNoise, WhiteNoise};
109 pub use super::process::*;
110 pub use super::simulator::TrackingArcSim;
111 pub use super::simulator::*;
112 pub use super::snc::*;
113 pub use super::*;
114
115 pub use crate::time::{Duration, Epoch, TimeUnits, Unit};
116}
117
118#[derive(Debug, PartialEq, Snafu)]
119#[snafu(visibility(pub(crate)))]
120pub enum ODError {
121 #[snafu(display("during an orbit determination, encountered {source}"))]
122 ODPropError {
123 #[snafu(source(from(PropagationError, Box::new)))]
124 source: Box<PropagationError>,
125 },
126 #[snafu(display("during an orbit determination, encountered {source}"))]
127 ODDynamicsError {
128 #[snafu(source(from(DynamicsError, Box::new)))]
129 source: Box<DynamicsError>,
130 },
131 #[snafu(display("at least {need} measurements required for {action}"))]
132 TooFewMeasurements { need: usize, action: &'static str },
133 #[snafu(display("invalid step size: {step}"))]
134 StepSizeError { step: Duration },
135 #[snafu(display("filter iteration did not converge in {loops} iterations"))]
136 Diverged { loops: usize },
137 #[snafu(display("STM is singular"))]
138 SingularStateTransitionMatrix,
139 #[snafu(display("invalid measurement @ {epoch} = {val}"))]
140 InvalidMeasurement { epoch: Epoch, val: f64 },
141 #[snafu(display("Kalman gain is singular"))]
142 SingularKalmanGain,
143 #[snafu(display("Information matrix is singular"))]
144 SingularInformationMatrix,
145 #[snafu(display("noise matrix is singular"))]
146 SingularNoiseRk,
147 #[snafu(display("{kind} noise not configured"))]
148 NoiseNotConfigured { kind: String },
149 #[snafu(display("measurement sim error: {details}"))]
150 MeasurementSimError { details: String },
151 #[snafu(display("during an OD encountered {source}: {details}"))]
152 ODTrajError { source: TrajError, details: String },
153 #[snafu(display("OD failed because {source}"))]
154 ODConfigError { source: ConfigError },
155 #[snafu(display("OD failed because of an I/O error: {source}"))]
156 ODIOError { source: InputOutputError },
157 #[snafu(display("OD failed due to Almanac: {action} {source}"))]
158 ODAlmanac {
159 #[snafu(source(from(AlmanacError, Box::new)))]
160 source: Box<AlmanacError>,
161 action: &'static str,
162 },
163 #[snafu(display("OD failed due to planetary data in Almanac: {action} {source}"))]
164 ODPlanetaryData {
165 #[snafu(source(from(PlanetaryDataError, Box::new)))]
166 source: Box<PlanetaryDataError>,
167 action: &'static str,
168 },
169 #[snafu(display("not enough residuals to {action}"))]
170 ODNoResiduals { action: &'static str },
171 #[snafu(display("could not {action} OD results: {source}"))]
172 ODStateError {
173 #[snafu(source(from(StateError, Box::new)))]
174 source: Box<StateError>,
175 action: &'static str,
176 },
177 #[snafu(display("Maximum iterations ({max_iter}) reached without convergence"))]
178 ODMaxIterations { max_iter: usize },
179 #[snafu(display("Nyx orbit determination limitation: {action}"))]
180 ODLimitation { action: &'static str },
181}