nyx_space/od/msr/measurement.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111
/*
Nyx, blazing fast astrodynamics
Copyright (C) 2018-onwards Christopher Rabotin <christopher.rabotin@gmail.com>
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
use super::MeasurementType;
use hifitime::Epoch;
use indexmap::{IndexMap, IndexSet};
use nalgebra::{allocator::Allocator, DefaultAllocator, DimName, OVector};
use std::fmt;
/// A type-agnostic simultaneous measurement storage structure. Allows storing any number of simultaneous measurement of a given taker.
///
/// Note that two measurements are considered equal if the tracker and epoch match exactly, and if both have the same measurement types,
/// and those measurements are equal to within 1e-10 (this allows for some leeway in TDM producers).
#[derive(Clone, Debug)]
pub struct Measurement {
/// Tracker alias which made this measurement
pub tracker: String,
/// Epoch of the measurement
pub epoch: Epoch,
/// All measurements made simultaneously
pub data: IndexMap<MeasurementType, f64>,
}
impl Measurement {
pub fn new(tracker: String, epoch: Epoch) -> Self {
Self {
tracker,
epoch,
data: IndexMap::new(),
}
}
pub fn push(&mut self, msr_type: MeasurementType, msr_value: f64) {
self.data.insert(msr_type, msr_value);
}
pub fn with(mut self, msr_type: MeasurementType, msr_value: f64) -> Self {
self.push(msr_type, msr_value);
self
}
/// Builds an observation vector for this measurement provided a set of measurement types.
/// If the requested measurement type is not available, then that specific row is set to zero.
/// The caller must set the appropriate sensitivity matrix rows to zero.
pub fn observation<S: DimName>(&self, types: &IndexSet<MeasurementType>) -> OVector<f64, S>
where
DefaultAllocator: Allocator<S>,
{
// Consider adding a modulo modifier here, any bias should be configured by each ground station.
let mut obs = OVector::zeros();
for (i, t) in types.iter().enumerate() {
if let Some(msr_value) = self.data.get(t) {
obs[i] = *msr_value;
}
}
obs
}
/// Returns a vector specifying which measurement types are available.
pub fn availability(&self, types: &IndexSet<MeasurementType>) -> Vec<bool> {
let mut rtn = vec![false; types.len()];
for (i, t) in types.iter().enumerate() {
if self.data.contains_key(t) {
rtn[i] = true;
}
}
rtn
}
}
impl fmt::Display for Measurement {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let msrs = self
.data
.iter()
.map(|(msr_type, msr_value)| format!("{msr_type:?} = {msr_value} {}", msr_type.unit()))
.collect::<Vec<String>>()
.join(", ");
write!(f, "{} measured {} on {}", self.tracker, msrs, self.epoch)
}
}
impl PartialEq for Measurement {
fn eq(&self, other: &Self) -> bool {
self.tracker == other.tracker
&& self.epoch == other.epoch
&& self.data.iter().all(|(key, &value)| {
if let Some(&other_value) = other.data.get(key) {
(value - other_value).abs() < 1e-10
} else {
false
}
})
}
}