sat-rs/satrs/src/mode_tree.rs

370 lines
11 KiB
Rust
Raw Normal View History

2024-02-21 15:37:08 +01:00
use std::sync::mpsc;
use alloc::vec::Vec;
use hashbrown::HashMap;
use crate::{
mode::{Mode, ModeAndSubmode, ModeReply, ModeRequest, Submode},
2024-02-22 15:36:55 +01:00
queue::GenericTargetedMessagingError,
request::{
2024-02-22 17:19:32 +01:00
MessageReceiver, MessageReceiverWithId, MessageSender, MessageSenderAndReceiver,
MessageSenderMap, MessageSenderMapWithId, MessageWithSenderId,
RequestAndReplySenderAndReceiver,
2024-02-22 15:36:55 +01:00
},
2024-02-21 15:37:08 +01:00
ChannelId,
};
#[derive(Debug, Copy, Clone, PartialEq, Eq)]
pub enum TableEntryType {
/// Target table containing information of the expected children modes for given mode.
Target,
/// Sequence table which contains information about how to reach a target table, including
/// the order of the sequences.
Sequence,
}
pub struct ModeTableEntry {
/// Name of respective table entry.
pub name: &'static str,
/// Target channel ID.
pub channel_id: ChannelId,
pub mode_submode: ModeAndSubmode,
pub allowed_submode_mask: Option<Submode>,
pub check_success: bool,
}
pub struct ModeTableMapValue {
/// Name for a given mode table entry.
pub name: &'static str,
pub entries: Vec<ModeTableEntry>,
}
pub type ModeTable = HashMap<Mode, ModeTableMapValue>;
2024-02-22 15:36:55 +01:00
pub trait ModeRequestSender {
fn local_channel_id(&self) -> ChannelId;
fn send_mode_request(
&self,
target_id: ChannelId,
request: ModeRequest,
) -> Result<(), GenericTargetedMessagingError>;
}
2024-02-22 15:36:55 +01:00
pub trait ModeReplySender {
2024-02-21 15:37:08 +01:00
fn local_channel_id(&self) -> ChannelId;
fn send_mode_reply(
2024-02-22 15:36:55 +01:00
&self,
2024-02-21 15:37:08 +01:00
target_id: ChannelId,
reply: ModeReply,
2024-02-22 15:36:55 +01:00
) -> Result<(), GenericTargetedMessagingError>;
2024-02-21 15:37:08 +01:00
}
2024-02-22 15:36:55 +01:00
pub trait ModeRequestReceiver {
fn try_recv_mode_request(
&self,
) -> Result<Option<MessageWithSenderId<ModeRequest>>, GenericTargetedMessagingError>;
2024-02-21 15:37:08 +01:00
}
2024-02-22 15:36:55 +01:00
pub trait ModeReplyReceiver {
fn try_recv_mode_reply(
&self,
) -> Result<Option<MessageWithSenderId<ModeReply>>, GenericTargetedMessagingError>;
}
2024-02-22 17:19:32 +01:00
impl<S: MessageSender<ModeRequest>> MessageSenderMap<ModeRequest, S> {
2024-02-22 15:36:55 +01:00
pub fn send_mode_request(
&self,
local_id: ChannelId,
target_id: ChannelId,
request: ModeRequest,
) -> Result<(), GenericTargetedMessagingError> {
self.send_message(local_id, target_id, request)
2024-02-21 15:37:08 +01:00
}
2024-02-22 17:19:32 +01:00
pub fn add_request_target(&mut self, target_id: ChannelId, request_sender: S) {
2024-02-22 15:36:55 +01:00
self.add_message_target(target_id, request_sender)
2024-02-21 15:37:08 +01:00
}
}
2024-02-22 17:19:32 +01:00
impl<S: MessageSender<ModeReply>> MessageSenderMap<ModeReply, S> {
2024-02-22 15:36:55 +01:00
pub fn send_mode_reply(
&self,
local_id: ChannelId,
target_id: ChannelId,
request: ModeReply,
) -> Result<(), GenericTargetedMessagingError> {
self.send_message(local_id, target_id, request)
}
2024-02-22 17:19:32 +01:00
pub fn add_reply_target(&mut self, target_id: ChannelId, request_sender: S) {
2024-02-22 15:36:55 +01:00
self.add_message_target(target_id, request_sender)
}
}
2024-02-22 17:19:32 +01:00
impl<S: MessageSender<ModeReply>> ModeReplySender for MessageSenderMapWithId<ModeReply, S> {
2024-02-22 15:36:55 +01:00
fn send_mode_reply(
&self,
target_channel_id: ChannelId,
reply: ModeReply,
) -> Result<(), GenericTargetedMessagingError> {
self.send_message(target_channel_id, reply)
}
2024-02-22 15:36:55 +01:00
fn local_channel_id(&self) -> ChannelId {
self.local_channel_id
}
}
2024-02-22 17:19:32 +01:00
impl<S: MessageSender<ModeRequest>> ModeRequestSender for MessageSenderMapWithId<ModeRequest, S> {
2024-02-22 15:36:55 +01:00
fn local_channel_id(&self) -> ChannelId {
self.local_channel_id
2024-02-21 19:16:25 +01:00
}
2024-02-22 15:36:55 +01:00
fn send_mode_request(
&self,
target_id: ChannelId,
request: ModeRequest,
) -> Result<(), GenericTargetedMessagingError> {
self.send_message(target_id, request)
2024-02-21 19:16:25 +01:00
}
}
2024-02-22 17:19:32 +01:00
impl<R: MessageReceiver<ModeReply>> ModeReplyReceiver for MessageReceiverWithId<ModeReply, R> {
2024-02-22 15:36:55 +01:00
fn try_recv_mode_reply(
&self,
) -> Result<Option<MessageWithSenderId<ModeReply>>, GenericTargetedMessagingError> {
self.try_recv_message()
2024-02-21 19:16:25 +01:00
}
}
2024-02-22 17:31:10 +01:00
2024-02-22 17:19:32 +01:00
impl<R: MessageReceiver<ModeRequest>> ModeRequestReceiver
for MessageReceiverWithId<ModeRequest, R>
{
2024-02-22 15:36:55 +01:00
fn try_recv_mode_request(
&self,
) -> Result<Option<MessageWithSenderId<ModeRequest>>, GenericTargetedMessagingError> {
self.try_recv_message()
}
2024-02-21 19:16:25 +01:00
}
2024-02-22 17:19:32 +01:00
impl<FROM, S: MessageSender<ModeRequest>, R: MessageReceiver<FROM>> ModeRequestSender
for MessageSenderAndReceiver<ModeRequest, FROM, S, R>
{
2024-02-22 15:36:55 +01:00
fn local_channel_id(&self) -> ChannelId {
self.local_channel_id_generic()
}
fn send_mode_request(
&self,
target_id: ChannelId,
request: ModeRequest,
) -> Result<(), GenericTargetedMessagingError> {
self.message_sender_map
.send_mode_request(self.local_channel_id(), target_id, request)
}
2024-02-21 19:16:25 +01:00
}
2024-02-22 17:19:32 +01:00
impl<FROM, S: MessageSender<ModeReply>, R: MessageReceiver<FROM>> ModeReplySender
for MessageSenderAndReceiver<ModeReply, FROM, S, R>
{
2024-02-22 15:36:55 +01:00
fn local_channel_id(&self) -> ChannelId {
self.local_channel_id_generic()
2024-02-21 19:16:25 +01:00
}
2024-02-22 15:36:55 +01:00
fn send_mode_reply(
&self,
target_id: ChannelId,
request: ModeReply,
) -> Result<(), GenericTargetedMessagingError> {
self.message_sender_map
.send_mode_reply(self.local_channel_id(), target_id, request)
}
}
2024-02-22 17:19:32 +01:00
impl<TO, S: MessageSender<TO>, R: MessageReceiver<ModeReply>> ModeReplyReceiver
for MessageSenderAndReceiver<TO, ModeReply, S, R>
{
2024-02-22 15:36:55 +01:00
fn try_recv_mode_reply(
&self,
) -> Result<Option<MessageWithSenderId<ModeReply>>, GenericTargetedMessagingError> {
self.message_receiver
.try_recv_message(self.local_channel_id_generic())
}
}
2024-02-22 17:19:32 +01:00
impl<TO, S: MessageSender<TO>, R: MessageReceiver<ModeRequest>> ModeRequestReceiver
for MessageSenderAndReceiver<TO, ModeRequest, S, R>
{
2024-02-22 15:36:55 +01:00
fn try_recv_mode_request(
2024-02-21 19:16:25 +01:00
&self,
2024-02-22 15:36:55 +01:00
) -> Result<Option<MessageWithSenderId<ModeRequest>>, GenericTargetedMessagingError> {
self.message_receiver
.try_recv_message(self.local_channel_id_generic())
}
}
2024-02-22 17:19:32 +01:00
pub type ModeRequestHandlerConnector<S, R> = MessageSenderAndReceiver<ModeReply, ModeRequest, S, R>;
pub type MpscModeRequestHandlerConnector = ModeRequestHandlerConnector<
mpsc::Sender<MessageWithSenderId<ModeReply>>,
mpsc::Receiver<MessageWithSenderId<ModeRequest>>,
>;
2024-02-22 17:31:10 +01:00
pub type MpscBoundedModeRequestHandlerConnector = ModeRequestHandlerConnector<
mpsc::SyncSender<MessageWithSenderId<ModeReply>>,
mpsc::Receiver<MessageWithSenderId<ModeRequest>>,
>;
2024-02-22 17:19:32 +01:00
pub type ModeRequestorConnector<S, R> = MessageSenderAndReceiver<ModeRequest, ModeReply, S, R>;
pub type MpscModeRequestorConnector = ModeRequestorConnector<
mpsc::Sender<MessageWithSenderId<ModeRequest>>,
mpsc::Receiver<MessageWithSenderId<ModeReply>>,
>;
2024-02-22 17:31:10 +01:00
pub type MpscBoundedModeRequestorConnector = ModeRequestorConnector<
mpsc::SyncSender<MessageWithSenderId<ModeRequest>>,
mpsc::Receiver<MessageWithSenderId<ModeReply>>,
>;
2024-02-22 17:19:32 +01:00
pub type ModeConnector<S0, R0, S1, R1> =
RequestAndReplySenderAndReceiver<ModeRequest, ModeReply, S0, R0, S1, R1>;
pub type MpscModeConnector = ModeConnector<
mpsc::Sender<MessageWithSenderId<ModeRequest>>,
mpsc::Receiver<MessageWithSenderId<ModeReply>>,
mpsc::Sender<MessageWithSenderId<ModeReply>>,
mpsc::Receiver<MessageWithSenderId<ModeRequest>>,
>;
2024-02-22 17:31:10 +01:00
pub type MpscBoundedModeConnector = ModeConnector<
mpsc::SyncSender<MessageWithSenderId<ModeRequest>>,
mpsc::Receiver<MessageWithSenderId<ModeReply>>,
mpsc::SyncSender<MessageWithSenderId<ModeReply>>,
mpsc::Receiver<MessageWithSenderId<ModeRequest>>,
>;
2024-02-22 17:19:32 +01:00
impl<
REPLY,
S0: MessageSender<ModeRequest>,
R0: MessageReceiver<REPLY>,
S1: MessageSender<REPLY>,
R1: MessageReceiver<ModeRequest>,
> RequestAndReplySenderAndReceiver<ModeRequest, REPLY, S0, R0, S1, R1>
{
pub fn add_request_target(&mut self, target_id: ChannelId, request_sender: S0) {
2024-02-22 15:36:55 +01:00
self.request_sender_map
.add_message_target(target_id, request_sender)
}
2024-02-21 15:37:08 +01:00
}
2024-02-22 17:19:32 +01:00
impl<
REQUEST,
S0: MessageSender<REQUEST>,
R0: MessageReceiver<ModeReply>,
S1: MessageSender<ModeReply>,
R1: MessageReceiver<REQUEST>,
> RequestAndReplySenderAndReceiver<REQUEST, ModeReply, S0, R0, S1, R1>
{
pub fn add_reply_target(&mut self, target_id: ChannelId, reply_sender: S1) {
2024-02-22 15:36:55 +01:00
self.reply_sender_map
.add_message_target(target_id, reply_sender)
}
2024-02-21 15:37:08 +01:00
}
2024-02-22 17:19:32 +01:00
impl<
REPLY,
S0: MessageSender<ModeRequest>,
R0: MessageReceiver<REPLY>,
S1: MessageSender<REPLY>,
R1: MessageReceiver<ModeRequest>,
> ModeRequestSender for RequestAndReplySenderAndReceiver<ModeRequest, REPLY, S0, R0, S1, R1>
{
2024-02-22 15:36:55 +01:00
fn local_channel_id(&self) -> ChannelId {
self.local_channel_id_generic()
}
2024-02-21 15:37:08 +01:00
fn send_mode_request(
2024-02-21 19:16:25 +01:00
&self,
2024-02-21 15:37:08 +01:00
target_id: ChannelId,
request: ModeRequest,
2024-02-22 15:36:55 +01:00
) -> Result<(), GenericTargetedMessagingError> {
self.request_sender_map
.send_mode_request(self.local_channel_id(), target_id, request)
}
}
2024-02-22 17:19:32 +01:00
impl<
REQUEST,
S0: MessageSender<REQUEST>,
R0: MessageReceiver<ModeReply>,
S1: MessageSender<ModeReply>,
R1: MessageReceiver<REQUEST>,
> ModeReplySender for RequestAndReplySenderAndReceiver<REQUEST, ModeReply, S0, R0, S1, R1>
{
2024-02-22 15:36:55 +01:00
fn local_channel_id(&self) -> ChannelId {
self.local_channel_id_generic()
2024-02-21 15:37:08 +01:00
}
2024-02-22 15:36:55 +01:00
fn send_mode_reply(
&self,
target_id: ChannelId,
request: ModeReply,
) -> Result<(), GenericTargetedMessagingError> {
self.reply_sender_map
.send_mode_reply(self.local_channel_id(), target_id, request)
2024-02-21 15:37:08 +01:00
}
}
2024-02-22 17:19:32 +01:00
impl<
REQUEST,
S0: MessageSender<REQUEST>,
R0: MessageReceiver<ModeReply>,
S1: MessageSender<ModeReply>,
R1: MessageReceiver<REQUEST>,
> ModeReplyReceiver for RequestAndReplySenderAndReceiver<REQUEST, ModeReply, S0, R0, S1, R1>
{
2024-02-22 15:36:55 +01:00
fn try_recv_mode_reply(
&self,
) -> Result<Option<MessageWithSenderId<ModeReply>>, GenericTargetedMessagingError> {
self.reply_receiver
.try_recv_message(self.local_channel_id_generic())
}
2024-02-22 15:36:55 +01:00
}
2024-02-22 17:19:32 +01:00
impl<
REPLY,
S0: MessageSender<ModeRequest>,
R0: MessageReceiver<REPLY>,
S1: MessageSender<REPLY>,
R1: MessageReceiver<ModeRequest>,
> ModeRequestReceiver for RequestAndReplySenderAndReceiver<ModeRequest, REPLY, S0, R0, S1, R1>
{
2024-02-22 15:36:55 +01:00
fn try_recv_mode_request(
&self,
) -> Result<Option<MessageWithSenderId<ModeRequest>>, GenericTargetedMessagingError> {
self.request_receiver
.try_recv_message(self.local_channel_id_generic())
}
}
2024-02-21 15:37:08 +01:00
pub trait ModeProvider {
fn mode_and_submode(&self) -> ModeAndSubmode;
}
#[derive(Debug, Clone)]
pub enum ModeError {
2024-02-22 15:36:55 +01:00
Messaging(GenericTargetedMessagingError),
}
impl From<GenericTargetedMessagingError> for ModeError {
fn from(value: GenericTargetedMessagingError) -> Self {
Self::Messaging(value)
}
2024-02-21 15:37:08 +01:00
}
pub trait ModeRequestHandler: ModeProvider {
fn start_transition(&mut self, mode_and_submode: ModeAndSubmode) -> Result<(), ModeError>;
fn announce_mode(&self, recursive: bool);
2024-02-22 15:36:55 +01:00
fn handle_mode_reached(&mut self) -> Result<(), GenericTargetedMessagingError>;
2024-02-21 15:37:08 +01:00
}
#[cfg(test)]
mod tests {
}