mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-06-28 19:07:26 +00:00
185 lines
5.0 KiB
Rust
185 lines
5.0 KiB
Rust
// Copyright 2017-2021 Parity Technologies (UK) Ltd.
|
|
// This file is part of Polkadot.
|
|
|
|
// Polkadot is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
|
|
// Polkadot 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 General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with Polkadot. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
//! A small set of wrapping types to cover most of our adversary test cases.
|
|
//!
|
|
//! This allows types with internal mutability to synchronize across
|
|
//! multiple subsystems and intercept or replace incoming and outgoing
|
|
//! messages on the overseer level.
|
|
|
|
use polkadot_node_subsystem::*;
|
|
pub use polkadot_node_subsystem::{messages::AllMessages, FromOverseer};
|
|
use std::future::Future;
|
|
use std::pin::Pin;
|
|
|
|
/// Filter incoming and outgoing messages.
|
|
pub trait MsgFilter: Send + Sync + Clone + 'static {
|
|
/// The message type the original subsystm handles incoming.
|
|
type Message: Send + 'static;
|
|
|
|
/// Filter messages that are to be received by
|
|
/// the subsystem.
|
|
fn filter_in(&self, msg: FromOverseer<Self::Message>) -> Option<FromOverseer<Self::Message>> {
|
|
Some(msg)
|
|
}
|
|
|
|
/// Modify outgoing messages.
|
|
fn filter_out(&self, msg: AllMessages) -> Option<AllMessages> {
|
|
Some(msg)
|
|
}
|
|
}
|
|
|
|
/// A sender with the outgoing messages filtered.
|
|
#[derive(Clone)]
|
|
pub struct FilteredSender<Sender, Fil> {
|
|
inner: Sender,
|
|
message_filter: Fil,
|
|
}
|
|
|
|
#[async_trait::async_trait]
|
|
impl<Sender, Fil> SubsystemSender for FilteredSender<Sender, Fil>
|
|
where
|
|
Sender: SubsystemSender,
|
|
Fil: MsgFilter,
|
|
{
|
|
async fn send_message(&mut self, msg: AllMessages) {
|
|
if let Some(msg) = self.message_filter.filter_out(msg) {
|
|
self.inner.send_message(msg).await;
|
|
}
|
|
}
|
|
|
|
async fn send_messages<T>(&mut self, msgs: T)
|
|
where
|
|
T: IntoIterator<Item = AllMessages> + Send,
|
|
T::IntoIter: Send,
|
|
{
|
|
for msg in msgs {
|
|
self.send_message(msg).await;
|
|
}
|
|
}
|
|
|
|
fn send_unbounded_message(&mut self, msg: AllMessages) {
|
|
if let Some(msg) = self.message_filter.filter_out(msg) {
|
|
self.inner.send_unbounded_message(msg);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// A subsystem context, that filters the outgoing messages.
|
|
pub struct FilteredContext<Context: SubsystemContext, Fil: MsgFilter> {
|
|
inner: Context,
|
|
message_filter: Fil,
|
|
sender: FilteredSender<<Context as SubsystemContext>::Sender, Fil>,
|
|
}
|
|
|
|
impl<Context, Fil> FilteredContext<Context, Fil>
|
|
where
|
|
Context: SubsystemContext,
|
|
Fil: MsgFilter<Message = <Context as SubsystemContext>::Message>,
|
|
{
|
|
pub fn new(mut inner: Context, message_filter: Fil) -> Self {
|
|
let sender = FilteredSender::<<Context as SubsystemContext>::Sender, Fil> {
|
|
inner: inner.sender().clone(),
|
|
message_filter: message_filter.clone(),
|
|
};
|
|
Self {
|
|
inner,
|
|
message_filter,
|
|
sender,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[async_trait::async_trait]
|
|
impl<Context, Fil> SubsystemContext for FilteredContext<Context, Fil>
|
|
where
|
|
Context: SubsystemContext,
|
|
Fil: MsgFilter<Message = <Context as SubsystemContext>::Message>,
|
|
{
|
|
type Message = <Context as SubsystemContext>::Message;
|
|
type Sender = FilteredSender<<Context as SubsystemContext>::Sender, Fil>;
|
|
|
|
async fn try_recv(&mut self) -> Result<Option<FromOverseer<Self::Message>>, ()> {
|
|
loop {
|
|
match self.inner.try_recv().await? {
|
|
None => return Ok(None),
|
|
Some(msg) => {
|
|
if let Some(msg) = self.message_filter.filter_in(msg) {
|
|
return Ok(Some(msg));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
async fn recv(&mut self) -> SubsystemResult<FromOverseer<Self::Message>> {
|
|
loop {
|
|
let msg = self.inner.recv().await?;
|
|
if let Some(msg) = self.message_filter.filter_in(msg) {
|
|
return Ok(msg);
|
|
}
|
|
}
|
|
}
|
|
|
|
async fn spawn(
|
|
&mut self,
|
|
name: &'static str,
|
|
s: Pin<Box<dyn Future<Output = ()> + Send>>,
|
|
) -> SubsystemResult<()> {
|
|
self.inner.spawn(name, s).await
|
|
}
|
|
|
|
async fn spawn_blocking(
|
|
&mut self,
|
|
name: &'static str,
|
|
s: Pin<Box<dyn Future<Output = ()> + Send>>,
|
|
) -> SubsystemResult<()> {
|
|
self.inner.spawn_blocking(name, s).await
|
|
}
|
|
|
|
fn sender(&mut self) -> &mut Self::Sender {
|
|
&mut self.sender
|
|
}
|
|
}
|
|
|
|
/// A subsystem to which incoming and outgoing filters are applied.
|
|
pub struct FilteredSubsystem<Sub, Fil> {
|
|
subsystem: Sub,
|
|
message_filter: Fil,
|
|
}
|
|
|
|
impl<Sub, Fil> FilteredSubsystem<Sub, Fil> {
|
|
pub fn new(subsystem: Sub, message_filter: Fil) -> Self {
|
|
Self {
|
|
subsystem,
|
|
message_filter,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<Context, Sub, Fil> Subsystem<Context> for FilteredSubsystem<Sub, Fil>
|
|
where
|
|
Context: SubsystemContext + Sync + Send,
|
|
Sub: Subsystem<FilteredContext<Context, Fil>>,
|
|
FilteredContext<Context, Fil>: SubsystemContext,
|
|
Fil: MsgFilter<Message = <Context as SubsystemContext>::Message>,
|
|
{
|
|
fn start(self, ctx: Context) -> SpawnedSubsystem {
|
|
let ctx = FilteredContext::new(ctx, self.message_filter);
|
|
Subsystem::<FilteredContext<Context, Fil>>::start(self.subsystem, ctx)
|
|
}
|
|
}
|