nyx_space/dynamics/guidance/
mod.rs1use crate::cosmic::{GuidanceMode, Orbit, Spacecraft, STD_GRAVITY};
20use crate::errors::{NyxError, StateError};
21use crate::linalg::Vector3;
22use anise::astro::PhysicsResult;
23use anise::errors::PhysicsError;
24use anise::math::rotation::DCM;
25use anise::prelude::Almanac;
26use der::{Decode, Encode, Reader};
27use serde::{Deserialize, Serialize};
28use serde_dhall::StaticType;
29
30pub mod mnvr;
31pub use mnvr::{Maneuver, MnvrRepr};
32
33mod replay;
34pub use replay::ThrustDirectionReplay;
35
36mod ruggiero;
37pub use ruggiero::{Objective, Ruggiero, StateParameter};
38
39mod kluever;
40pub use kluever::Kluever;
41use snafu::Snafu;
42
43use std::fmt;
44use std::sync::Arc;
45
46#[cfg(feature = "python")]
47use pyo3::prelude::*;
48
49#[allow(non_snake_case)]
51#[cfg_attr(feature = "python", pyclass(get_all, set_all))]
52#[derive(Copy, Clone, Debug, PartialEq, Serialize, Deserialize, StaticType)]
53pub struct Thruster {
54 pub thrust_N: f64,
56 pub isp_s: f64,
58}
59
60#[cfg_attr(feature = "python", pymethods)]
61impl Thruster {
62 pub fn exhaust_velocity_m_s(&self) -> f64 {
64 self.isp_s * STD_GRAVITY
65 }
66}
67
68#[cfg(feature = "python")]
69#[cfg_attr(feature = "python", pymethods)]
70impl Thruster {
71 #[allow(non_snake_case)]
72 #[new]
73 fn py_new(thrust_N: f64, isp_s: f64) -> Self {
74 Self { thrust_N, isp_s }
75 }
76}
77
78impl Encode for Thruster {
79 fn encoded_len(&self) -> der::Result<der::Length> {
80 self.thrust_N.encoded_len()? + self.isp_s.encoded_len()?
81 }
82
83 fn encode(&self, encoder: &mut impl der::Writer) -> der::Result<()> {
84 self.thrust_N.encode(encoder)?;
85 self.isp_s.encode(encoder)
86 }
87}
88
89impl<'a> Decode<'a> for Thruster {
90 fn decode<R: Reader<'a>>(decoder: &mut R) -> der::Result<Self> {
91 Ok(Self {
92 thrust_N: decoder.decode()?,
93 isp_s: decoder.decode()?,
94 })
95 }
96}
97
98#[derive(Clone, Debug, Serialize, Deserialize, StaticType)]
99pub struct ObjectiveEfficiency {
100 pub objective: Objective,
101 pub efficiency: f64,
102}
103
104#[derive(Clone, Debug, Serialize, Deserialize, StaticType)]
105pub struct ObjectiveWeight {
106 pub objective: Objective,
107 pub weight: f64,
108}
109
110pub trait GuidanceLaw: fmt::Display + Send + Sync {
114 fn direction(&self, osc_state: &Spacecraft) -> Result<Vector3<f64>, GuidanceError>;
116
117 fn throttle(&self, osc_state: &Spacecraft) -> Result<f64, GuidanceError>;
120
121 fn next(&self, next_state: &mut Spacecraft, almanac: Arc<Almanac>);
123
124 fn achieved(&self, _osc_state: &Spacecraft) -> Result<bool, GuidanceError> {
126 Err(GuidanceError::NoGuidanceObjectiveDefined)
127 }
128}
129
130fn unit_vector_from_plane_angles(alpha: f64, beta: f64) -> Vector3<f64> {
132 Vector3::new(
133 alpha.sin() * beta.cos(),
134 alpha.cos() * beta.cos(),
135 beta.sin(),
136 )
137}
138
139pub fn plane_angles_from_unit_vector(vhat: Vector3<f64>) -> (f64, f64) {
141 (vhat[1].atan2(vhat[0]), vhat[2].asin())
142}
143
144pub(crate) fn unit_vector_from_ra_dec(alpha: f64, delta: f64) -> Vector3<f64> {
146 Vector3::new(
147 delta.cos() * alpha.cos(),
148 delta.cos() * alpha.sin(),
149 delta.sin(),
150 )
151}
152
153pub(crate) fn ra_dec_from_unit_vector(vhat: Vector3<f64>) -> (f64, f64) {
155 let alpha = vhat[1].atan2(vhat[0]);
156 let delta = vhat[2].asin();
157 (alpha, delta)
158}
159
160#[derive(Debug, PartialEq, Snafu)]
161pub enum GuidanceError {
162 #[snafu(display("No thruster attached to spacecraft"))]
163 NoThrustersDefined,
164 #[snafu(display("Throttle is not between 0.0 and 1.0: {ratio}"))]
165 ThrottleRatio { ratio: f64 },
166 #[snafu(display("Invalid finite burn control direction u = [{x}, {y}, {z}] => i-plane = {in_plane_deg} deg, Delta = {out_of_plane_deg} deg",))]
167 InvalidDirection {
168 x: f64,
169 y: f64,
170 z: f64,
171 in_plane_deg: f64,
172 out_of_plane_deg: f64,
173 },
174 #[snafu(display("Invalid finite burn control rate u = [{x}, {y}, {z}] => in-plane = {in_plane_deg_s} deg/s, out of plane = {out_of_plane_deg_s} deg/s",))]
175 InvalidRate {
176 x: f64,
177 y: f64,
178 z: f64,
179 in_plane_deg_s: f64,
180 out_of_plane_deg_s: f64,
181 },
182 #[snafu(display("Invalid finite burn control acceleration u = [{x}, {y}, {z}] => in-plane = {in_plane_deg_s2} deg/s^2, out of plane = {out_of_plane_deg_s2} deg/s^2",))]
183 InvalidAcceleration {
184 x: f64,
185 y: f64,
186 z: f64,
187 in_plane_deg_s2: f64,
188 out_of_plane_deg_s2: f64,
189 },
190 #[snafu(display("when {action} encountered {source}"))]
191 GuidancePhysicsError {
192 action: &'static str,
193 source: PhysicsError,
194 },
195 #[snafu(display(
196 "An objective based analysis or control was attempted, but no objective was defined"
197 ))]
198 NoGuidanceObjectiveDefined,
199 #[snafu(display("{param} is not a control variable in this guidance law"))]
200 InvalidControl { param: StateParameter },
201 #[snafu(display("guidance encountered {source}"))]
202 GuidState { source: StateError },
203}
204
205#[derive(Copy, Clone, Debug, PartialEq, Eq, Serialize, Deserialize, StaticType)]
208pub enum LocalFrame {
209 Inertial,
210 RIC,
211 VNC,
212 RCN,
213}
214
215impl LocalFrame {
216 pub fn dcm_to_inertial(&self, state: Orbit) -> PhysicsResult<DCM> {
217 match self {
218 LocalFrame::Inertial => Ok(DCM::identity(
219 state.frame.orientation_id,
220 state.frame.orientation_id,
221 )),
222 LocalFrame::RIC => state.dcm_from_ric_to_inertial(),
223 LocalFrame::VNC => state.dcm_from_vnc_to_inertial(),
224 LocalFrame::RCN => state.dcm_from_rcn_to_inertial(),
225 }
226 }
227}
228
229#[test]
230fn ra_dec_from_vec() {
231 use std::f64::consts::{FRAC_PI_2, PI, TAU};
232 let mut delta = -FRAC_PI_2;
233 let mut alpha = 0.0;
234 loop {
235 loop {
236 let unit_v = unit_vector_from_ra_dec(alpha, delta);
237 let (alpha2, delta2) = ra_dec_from_unit_vector(unit_v);
238 assert!((alpha - alpha2).abs() < f64::EPSILON);
239 assert!((delta - delta2).abs() < f64::EPSILON);
240 alpha += TAU * 0.1; if alpha > PI {
242 alpha = 0.0;
243 break;
244 }
245 }
246 delta += TAU * 0.1; if delta > FRAC_PI_2 {
248 break;
249 }
250 }
251}