use async_io::Timer;
use std::time::Duration;
use web_time::Instant;
use crate::runtime::BlockMeta;
use crate::runtime::Kernel;
use crate::runtime::MessageOutputs;
use crate::runtime::Pmt;
use crate::runtime::Result;
use crate::runtime::WorkIo;
#[derive(Block)]
#[message_outputs(out)]
pub struct MessageSource {
message: Pmt,
interval: Duration,
t_last: Instant,
n_messages: Option<usize>,
}
impl MessageSource {
pub fn new(message: Pmt, interval: Duration, n_messages: Option<usize>) -> Self {
Self {
message,
interval,
t_last: Instant::now(),
n_messages,
}
}
async fn sleep(dur: Duration) {
Timer::after(dur).await;
}
}
#[doc(hidden)]
impl Kernel for MessageSource {
async fn work(
&mut self,
io: &mut WorkIo,
mio: &mut MessageOutputs,
_b: &mut BlockMeta,
) -> Result<()> {
let now = Instant::now();
if now >= self.t_last + self.interval {
mio.post("out", self.message.clone()).await?;
self.t_last = now;
if let Some(ref mut n) = self.n_messages {
*n -= 1;
if *n == 0 {
io.finished = true;
}
}
}
io.block_on(MessageSource::sleep(
self.t_last + self.interval - Instant::now(),
));
Ok(())
}
async fn init(&mut self, _mio: &mut MessageOutputs, _b: &mut BlockMeta) -> Result<()> {
self.t_last = Instant::now();
Ok(())
}
}
pub struct MessageSourceBuilder {
message: Pmt,
duration: Duration,
n_messages: Option<usize>,
}
impl MessageSourceBuilder {
pub fn new(message: Pmt, duration: Duration) -> MessageSourceBuilder {
MessageSourceBuilder {
message,
duration,
n_messages: None,
}
}
#[must_use]
pub fn n_messages(mut self, n: usize) -> MessageSourceBuilder {
self.n_messages = Some(n);
self
}
pub fn build(self) -> MessageSource {
MessageSource::new(self.message, self.duration, self.n_messages)
}
}