nyx_space/od/ground_station/
mod.rsuse anise::astro::{Aberration, AzElRange, PhysicsResult};
use anise::constants::frames::EARTH_J2000;
use anise::errors::AlmanacResult;
use anise::prelude::{Almanac, Frame, Orbit};
use indexmap::{IndexMap, IndexSet};
use snafu::ensure;
use super::msr::MeasurementType;
use super::noise::{GaussMarkov, StochasticNoise};
use super::{ODAlmanacSnafu, ODError, ODTrajSnafu, TrackingDevice};
use crate::io::ConfigRepr;
use crate::od::NoiseNotConfiguredSnafu;
use crate::time::Epoch;
use hifitime::Duration;
use rand_pcg::Pcg64Mcg;
use serde_derive::{Deserialize, Serialize};
use std::fmt;
pub mod builtin;
pub mod event;
pub mod trk_device;
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)]
pub struct GroundStation {
pub name: String,
pub elevation_mask_deg: f64,
pub latitude_deg: f64,
pub longitude_deg: f64,
pub height_km: f64,
pub frame: Frame,
pub measurement_types: IndexSet<MeasurementType>,
pub integration_time: Option<Duration>,
pub light_time_correction: bool,
pub timestamp_noise_s: Option<StochasticNoise>,
pub stochastic_noises: Option<IndexMap<MeasurementType, StochasticNoise>>,
}
impl GroundStation {
pub fn from_point(
name: String,
latitude_deg: f64,
longitude_deg: f64,
height_km: f64,
frame: Frame,
) -> Self {
Self {
name,
elevation_mask_deg: 0.0,
latitude_deg,
longitude_deg,
height_km,
frame,
measurement_types: IndexSet::new(),
integration_time: None,
light_time_correction: false,
timestamp_noise_s: None,
stochastic_noises: None,
}
}
pub fn with_msr_type(mut self, msr_type: MeasurementType, noise: StochasticNoise) -> Self {
if self.stochastic_noises.is_none() {
self.stochastic_noises = Some(IndexMap::new());
}
self.stochastic_noises
.as_mut()
.unwrap()
.insert(msr_type, noise);
self.measurement_types.insert(msr_type);
self
}
pub fn without_msr_type(mut self, msr_type: MeasurementType) -> Self {
if let Some(noises) = self.stochastic_noises.as_mut() {
noises.swap_remove(&msr_type);
}
self.measurement_types.swap_remove(&msr_type);
self
}
pub fn with_integration_time(mut self, integration_time: Option<Duration>) -> Self {
self.integration_time = integration_time;
self
}
pub fn with_msr_bias_constant(
mut self,
msr_type: MeasurementType,
bias_constant: f64,
) -> Result<Self, ODError> {
if self.stochastic_noises.is_none() {
self.stochastic_noises = Some(IndexMap::new());
}
let stochastics = self.stochastic_noises.as_mut().unwrap();
let this_noise = stochastics
.get_mut(&msr_type)
.ok_or(ODError::NoiseNotConfigured {
kind: format!("{msr_type:?}"),
})
.unwrap();
if this_noise.bias.is_none() {
this_noise.bias = Some(GaussMarkov::ZERO);
}
this_noise.bias.unwrap().constant = Some(bias_constant);
Ok(self)
}
pub fn azimuth_elevation_of(
&self,
rx: Orbit,
obstructing_body: Option<Frame>,
almanac: &Almanac,
) -> AlmanacResult<AzElRange> {
let ab_corr = if self.light_time_correction {
Aberration::LT
} else {
Aberration::NONE
};
almanac.azimuth_elevation_range_sez(
rx,
self.to_orbit(rx.epoch, almanac).unwrap(),
obstructing_body,
ab_corr,
)
}
pub fn to_orbit(&self, epoch: Epoch, almanac: &Almanac) -> PhysicsResult<Orbit> {
use anise::constants::usual_planetary_constants::MEAN_EARTH_ANGULAR_VELOCITY_DEG_S;
Orbit::try_latlongalt(
self.latitude_deg,
self.longitude_deg,
self.height_km,
MEAN_EARTH_ANGULAR_VELOCITY_DEG_S,
epoch,
almanac.frame_from_uid(self.frame).unwrap(),
)
}
fn noises(&mut self, epoch: Epoch, rng: Option<&mut Pcg64Mcg>) -> Result<Vec<f64>, ODError> {
let mut noises = vec![0.0; self.measurement_types.len() + 1];
if let Some(rng) = rng {
ensure!(
self.stochastic_noises.is_some(),
NoiseNotConfiguredSnafu {
kind: "ground station stochastics".to_string(),
}
);
if let Some(mut timestamp_noise) = self.timestamp_noise_s {
noises[0] = timestamp_noise.sample(epoch, rng);
}
let stochastics = self.stochastic_noises.as_mut().unwrap();
for (ii, msr_type) in self.measurement_types.iter().enumerate() {
noises[ii + 1] = stochastics
.get_mut(msr_type)
.ok_or(ODError::NoiseNotConfigured {
kind: format!("{msr_type:?}"),
})?
.sample(epoch, rng);
}
}
Ok(noises)
}
}
impl Default for GroundStation {
fn default() -> Self {
let mut measurement_types = IndexSet::new();
measurement_types.insert(MeasurementType::Range);
measurement_types.insert(MeasurementType::Doppler);
Self {
name: "UNDEFINED".to_string(),
measurement_types,
elevation_mask_deg: 0.0,
latitude_deg: 0.0,
longitude_deg: 0.0,
height_km: 0.0,
frame: EARTH_J2000,
integration_time: None,
light_time_correction: false,
timestamp_noise_s: None,
stochastic_noises: None,
}
}
}
impl ConfigRepr for GroundStation {}
impl fmt::Display for GroundStation {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(
f,
"{} (lat.: {:.4} deg long.: {:.4} deg alt.: {:.3} m) [{}]",
self.name,
self.latitude_deg,
self.longitude_deg,
self.height_km * 1e3,
self.frame,
)
}
}
#[cfg(test)]
mod gs_ut {
use anise::constants::frames::IAU_EARTH_FRAME;
use indexmap::{IndexMap, IndexSet};
use crate::io::ConfigRepr;
use crate::od::prelude::*;
#[test]
fn test_load_single() {
use std::env;
use std::path::PathBuf;
use hifitime::TimeUnits;
let test_data: PathBuf = [
env::var("CARGO_MANIFEST_DIR").unwrap(),
"data".to_string(),
"tests".to_string(),
"config".to_string(),
"one_ground_station.yaml".to_string(),
]
.iter()
.collect();
assert!(test_data.exists(), "Could not find the test data");
let gs = GroundStation::load(test_data).unwrap();
dbg!(&gs);
let mut measurement_types = IndexSet::new();
measurement_types.insert(MeasurementType::Range);
measurement_types.insert(MeasurementType::Doppler);
let mut stochastics = IndexMap::new();
stochastics.insert(
MeasurementType::Range,
StochasticNoise {
bias: Some(GaussMarkov::new(1.days(), 5e-3).unwrap()),
..Default::default()
},
);
stochastics.insert(
MeasurementType::Doppler,
StochasticNoise {
bias: Some(GaussMarkov::new(1.days(), 5e-5).unwrap()),
..Default::default()
},
);
let expected_gs = GroundStation {
name: "Demo ground station".to_string(),
frame: IAU_EARTH_FRAME,
measurement_types,
elevation_mask_deg: 5.0,
stochastic_noises: Some(stochastics),
latitude_deg: 2.3522,
longitude_deg: 48.8566,
height_km: 0.4,
light_time_correction: false,
timestamp_noise_s: None,
integration_time: Some(60 * Unit::Second),
};
println!("{}", serde_yml::to_string(&expected_gs).unwrap());
assert_eq!(expected_gs, gs);
}
#[test]
fn test_load_many() {
use hifitime::TimeUnits;
use std::env;
use std::path::PathBuf;
let test_file: PathBuf = [
env::var("CARGO_MANIFEST_DIR").unwrap(),
"data".to_string(),
"tests".to_string(),
"config".to_string(),
"many_ground_stations.yaml".to_string(),
]
.iter()
.collect();
let stations = GroundStation::load_many(test_file).unwrap();
dbg!(&stations);
let mut measurement_types = IndexSet::new();
measurement_types.insert(MeasurementType::Range);
measurement_types.insert(MeasurementType::Doppler);
let mut stochastics = IndexMap::new();
stochastics.insert(
MeasurementType::Range,
StochasticNoise {
bias: Some(GaussMarkov::new(1.days(), 5e-3).unwrap()),
..Default::default()
},
);
stochastics.insert(
MeasurementType::Doppler,
StochasticNoise {
bias: Some(GaussMarkov::new(1.days(), 5e-5).unwrap()),
..Default::default()
},
);
let expected = vec![
GroundStation {
name: "Demo ground station".to_string(),
frame: IAU_EARTH_FRAME.with_mu_km3_s2(398600.435436096),
measurement_types: measurement_types.clone(),
elevation_mask_deg: 5.0,
stochastic_noises: Some(stochastics.clone()),
latitude_deg: 2.3522,
longitude_deg: 48.8566,
height_km: 0.4,
light_time_correction: false,
timestamp_noise_s: None,
integration_time: None,
},
GroundStation {
name: "Canberra".to_string(),
frame: IAU_EARTH_FRAME.with_mu_km3_s2(398600.435436096),
measurement_types,
elevation_mask_deg: 5.0,
stochastic_noises: Some(stochastics),
latitude_deg: -35.398333,
longitude_deg: 148.981944,
height_km: 0.691750,
light_time_correction: false,
timestamp_noise_s: None,
integration_time: None,
},
];
assert_eq!(expected, stations);
let reser = serde_yml::to_string(&expected).unwrap();
dbg!(reser);
}
}