2024-04-09 11:18:54 +02:00
|
|
|
use std::collections::HashMap;
|
|
|
|
use std::sync::mpsc;
|
|
|
|
|
|
|
|
use log::warn;
|
2024-04-09 13:52:02 +02:00
|
|
|
use ops_sat_rs::config::components::PUS_ROUTING_SERVICE;
|
|
|
|
use ops_sat_rs::config::tmtc_err;
|
2024-04-09 11:18:54 +02:00
|
|
|
use satrs::action::ActionRequest;
|
|
|
|
use satrs::hk::HkRequest;
|
|
|
|
use satrs::mode::ModeRequest;
|
|
|
|
use satrs::pus::verification::{
|
|
|
|
FailParams, TcStateAccepted, VerificationReportingProvider, VerificationToken,
|
|
|
|
};
|
2024-04-15 12:16:01 +02:00
|
|
|
use satrs::pus::{ActiveRequestProvider, EcssTmSender, GenericRoutingError, PusRequestRouter};
|
2024-04-09 11:18:54 +02:00
|
|
|
use satrs::queue::GenericSendError;
|
|
|
|
use satrs::request::{GenericMessage, MessageMetadata, UniqueApidTargetId};
|
|
|
|
use satrs::spacepackets::ecss::tc::PusTcReader;
|
|
|
|
use satrs::spacepackets::ecss::PusPacket;
|
|
|
|
use satrs::ComponentId;
|
|
|
|
|
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
#[non_exhaustive]
|
|
|
|
pub enum CompositeRequest {
|
|
|
|
Hk(HkRequest),
|
|
|
|
Action(ActionRequest),
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone)]
|
|
|
|
pub struct GenericRequestRouter {
|
|
|
|
pub id: ComponentId,
|
|
|
|
// All messages which do not have a dedicated queue.
|
|
|
|
pub composite_router_map: HashMap<ComponentId, mpsc::Sender<GenericMessage<CompositeRequest>>>,
|
|
|
|
pub mode_router_map: HashMap<ComponentId, mpsc::Sender<GenericMessage<ModeRequest>>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for GenericRequestRouter {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
|
|
|
id: PUS_ROUTING_SERVICE.raw(),
|
|
|
|
composite_router_map: Default::default(),
|
|
|
|
mode_router_map: Default::default(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-04-09 13:52:02 +02:00
|
|
|
|
|
|
|
#[allow(dead_code)]
|
2024-04-09 11:18:54 +02:00
|
|
|
impl GenericRequestRouter {
|
|
|
|
pub(crate) fn handle_error_generic(
|
|
|
|
&self,
|
|
|
|
active_request: &impl ActiveRequestProvider,
|
|
|
|
tc: &PusTcReader,
|
|
|
|
error: GenericRoutingError,
|
2024-04-15 12:16:01 +02:00
|
|
|
tm_sender: &(impl EcssTmSender + ?Sized),
|
2024-04-09 11:18:54 +02:00
|
|
|
verif_reporter: &impl VerificationReportingProvider,
|
|
|
|
time_stamp: &[u8],
|
|
|
|
) {
|
|
|
|
warn!(
|
|
|
|
"Routing request for service {} failed: {error:?}",
|
|
|
|
tc.service()
|
|
|
|
);
|
|
|
|
let accepted_token: VerificationToken<TcStateAccepted> = active_request
|
|
|
|
.token()
|
|
|
|
.try_into()
|
|
|
|
.expect("token is not in accepted state");
|
|
|
|
match error {
|
|
|
|
GenericRoutingError::UnknownTargetId(id) => {
|
|
|
|
let apid_target_id = UniqueApidTargetId::from(id);
|
|
|
|
warn!("Target APID for request: {}", apid_target_id.apid);
|
|
|
|
warn!("Target Unique ID for request: {}", apid_target_id.unique_id);
|
|
|
|
let mut fail_data: [u8; 8] = [0; 8];
|
|
|
|
fail_data.copy_from_slice(&id.to_be_bytes());
|
|
|
|
verif_reporter
|
|
|
|
.completion_failure(
|
|
|
|
tm_sender,
|
|
|
|
accepted_token,
|
|
|
|
FailParams::new(time_stamp, &tmtc_err::UNKNOWN_TARGET_ID, &fail_data),
|
|
|
|
)
|
|
|
|
.expect("Sending start failure failed");
|
|
|
|
}
|
|
|
|
GenericRoutingError::Send(_) => {
|
|
|
|
let mut fail_data: [u8; 8] = [0; 8];
|
|
|
|
fail_data.copy_from_slice(&active_request.target_id().to_be_bytes());
|
|
|
|
verif_reporter
|
|
|
|
.completion_failure(
|
|
|
|
tm_sender,
|
|
|
|
accepted_token,
|
|
|
|
FailParams::new(time_stamp, &tmtc_err::ROUTING_ERROR, &fail_data),
|
|
|
|
)
|
|
|
|
.expect("Sending start failure failed");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl PusRequestRouter<HkRequest> for GenericRequestRouter {
|
|
|
|
type Error = GenericRoutingError;
|
|
|
|
|
|
|
|
fn route(
|
|
|
|
&self,
|
|
|
|
requestor_info: MessageMetadata,
|
|
|
|
target_id: ComponentId,
|
|
|
|
hk_request: HkRequest,
|
|
|
|
) -> Result<(), Self::Error> {
|
|
|
|
if let Some(sender) = self.composite_router_map.get(&target_id) {
|
|
|
|
sender
|
|
|
|
.send(GenericMessage::new(
|
|
|
|
requestor_info,
|
|
|
|
CompositeRequest::Hk(hk_request),
|
|
|
|
))
|
|
|
|
.map_err(|_| GenericRoutingError::Send(GenericSendError::RxDisconnected))?;
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
Err(GenericRoutingError::UnknownTargetId(target_id))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PusRequestRouter<ActionRequest> for GenericRequestRouter {
|
|
|
|
type Error = GenericRoutingError;
|
|
|
|
|
|
|
|
fn route(
|
|
|
|
&self,
|
|
|
|
requestor_info: MessageMetadata,
|
|
|
|
target_id: ComponentId,
|
|
|
|
action_request: ActionRequest,
|
|
|
|
) -> Result<(), Self::Error> {
|
|
|
|
if let Some(sender) = self.composite_router_map.get(&target_id) {
|
|
|
|
sender
|
|
|
|
.send(GenericMessage::new(
|
|
|
|
requestor_info,
|
|
|
|
CompositeRequest::Action(action_request),
|
|
|
|
))
|
|
|
|
.map_err(|_| GenericRoutingError::Send(GenericSendError::RxDisconnected))?;
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
Err(GenericRoutingError::UnknownTargetId(target_id))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PusRequestRouter<ModeRequest> for GenericRequestRouter {
|
|
|
|
type Error = GenericRoutingError;
|
|
|
|
|
|
|
|
fn route(
|
|
|
|
&self,
|
|
|
|
requestor_info: MessageMetadata,
|
|
|
|
target_id: ComponentId,
|
|
|
|
request: ModeRequest,
|
|
|
|
) -> Result<(), Self::Error> {
|
|
|
|
if let Some(sender) = self.mode_router_map.get(&target_id) {
|
|
|
|
sender
|
|
|
|
.send(GenericMessage::new(requestor_info, request))
|
|
|
|
.map_err(|_| GenericRoutingError::Send(GenericSendError::RxDisconnected))?;
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
Err(GenericRoutingError::UnknownTargetId(target_id))
|
|
|
|
}
|
|
|
|
}
|