|
|
@ -1,27 +1,18 @@ |
|
|
|
use crate::model::{ProgramId, ProgramRef, Programs, Sections}; |
|
|
|
use crate::model::{ProgramId, ProgramRef, Programs, Sections}; |
|
|
|
use crate::section_runner::{SectionEvent, SectionRunHandle, SectionRunner}; |
|
|
|
use crate::section_runner::{ |
|
|
|
use eyre::WrapErr; |
|
|
|
Error as SectionRunnerError, SectionEvent, SectionEventRecv, SectionRunHandle, SectionRunner, |
|
|
|
|
|
|
|
}; |
|
|
|
|
|
|
|
use actix::{ |
|
|
|
|
|
|
|
Actor, ActorContext, ActorFuture, ActorStream, Addr, AsyncContext, Handler, Message, |
|
|
|
|
|
|
|
MessageResult, SpawnHandle, StreamHandler, WrapFuture, |
|
|
|
|
|
|
|
}; |
|
|
|
use std::collections::VecDeque; |
|
|
|
use std::collections::VecDeque; |
|
|
|
use thiserror::Error; |
|
|
|
use thiserror::Error; |
|
|
|
use tokio::{ |
|
|
|
use tokio::{ |
|
|
|
spawn, |
|
|
|
|
|
|
|
stream::StreamExt, |
|
|
|
|
|
|
|
sync::{broadcast, mpsc, oneshot}, |
|
|
|
sync::{broadcast, mpsc, oneshot}, |
|
|
|
time::{delay_queue, DelayQueue}, |
|
|
|
time::{delay_queue, DelayQueue}, |
|
|
|
}; |
|
|
|
}; |
|
|
|
use tracing::{debug, error, trace, trace_span, warn}; |
|
|
|
use tracing::{debug, error, trace, warn}; |
|
|
|
use tracing_futures::Instrument; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#[derive(Debug)] |
|
|
|
|
|
|
|
enum RunnerMsg { |
|
|
|
|
|
|
|
Quit(oneshot::Sender<()>), |
|
|
|
|
|
|
|
UpdateSections(Sections), |
|
|
|
|
|
|
|
UpdatePrograms(Programs), |
|
|
|
|
|
|
|
RunProgramId(ProgramId), |
|
|
|
|
|
|
|
RunProgram(ProgramRef), |
|
|
|
|
|
|
|
CancelProgram(ProgramId), |
|
|
|
|
|
|
|
Subscribe(oneshot::Sender<ProgramEventRecv>), |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#[derive(Clone, Debug)] |
|
|
|
#[derive(Clone, Debug)] |
|
|
|
pub enum ProgramEvent { |
|
|
|
pub enum ProgramEvent { |
|
|
@ -62,31 +53,15 @@ impl ProgRun { |
|
|
|
|
|
|
|
|
|
|
|
type RunQueue = VecDeque<ProgRun>; |
|
|
|
type RunQueue = VecDeque<ProgRun>; |
|
|
|
|
|
|
|
|
|
|
|
struct RunnerTask { |
|
|
|
struct ProgramRunnerInner { |
|
|
|
section_runner: SectionRunner, |
|
|
|
section_runner: SectionRunner, |
|
|
|
msg_recv: mpsc::Receiver<RunnerMsg>, |
|
|
|
|
|
|
|
running: bool, |
|
|
|
|
|
|
|
sections: Sections, |
|
|
|
sections: Sections, |
|
|
|
programs: Programs, |
|
|
|
programs: Programs, |
|
|
|
event_send: Option<ProgramEventSend>, |
|
|
|
event_send: Option<ProgramEventSend>, |
|
|
|
scheduled_run_queue: DelayQueue<ProgramRef>, |
|
|
|
schedule_run_fut: Option<SpawnHandle>, |
|
|
|
quit_tx: Option<oneshot::Sender<()>>, |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
impl RunnerTask { |
|
|
|
impl ProgramRunnerInner { |
|
|
|
fn new(section_runner: SectionRunner, msg_recv: mpsc::Receiver<RunnerMsg>) -> Self { |
|
|
|
|
|
|
|
Self { |
|
|
|
|
|
|
|
section_runner, |
|
|
|
|
|
|
|
msg_recv, |
|
|
|
|
|
|
|
running: true, |
|
|
|
|
|
|
|
sections: Sections::new(), |
|
|
|
|
|
|
|
programs: Programs::new(), |
|
|
|
|
|
|
|
event_send: None, |
|
|
|
|
|
|
|
scheduled_run_queue: DelayQueue::new(), |
|
|
|
|
|
|
|
quit_tx: None, |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
fn send_event(&mut self, event: ProgramEvent) { |
|
|
|
fn send_event(&mut self, event: ProgramEvent) { |
|
|
|
if let Some(event_send) = &mut self.event_send { |
|
|
|
if let Some(event_send) = &mut self.event_send { |
|
|
|
match event_send.send(event) { |
|
|
|
match event_send.send(event) { |
|
|
@ -109,88 +84,57 @@ impl RunnerTask { |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
async fn start_program_run(&mut self, run: &mut ProgRun) -> eyre::Result<()> { |
|
|
|
fn start_program_run(&mut self, run: &mut ProgRun) { |
|
|
|
if run.state != RunState::Waiting { |
|
|
|
if run.state != RunState::Waiting { |
|
|
|
warn!( |
|
|
|
warn!( |
|
|
|
program_id = run.program.id, |
|
|
|
program_id = run.program.id, |
|
|
|
"cannot run program which is already running" |
|
|
|
"cannot run program which is already running" |
|
|
|
); |
|
|
|
); |
|
|
|
return Ok(()); |
|
|
|
return; |
|
|
|
} |
|
|
|
} |
|
|
|
run.sec_run_handles.reserve(run.program.sequence.len()); |
|
|
|
let sequence: Vec<_> = run |
|
|
|
for item in &run.program.sequence { |
|
|
|
.program |
|
|
|
let section = match self.sections.get(&item.section_id) { |
|
|
|
.sequence |
|
|
|
Some(sec) => sec.clone(), |
|
|
|
.iter() |
|
|
|
|
|
|
|
.filter_map(|item| match self.sections.get(&item.section_id) { |
|
|
|
|
|
|
|
Some(sec) => Some((sec.clone(), item.duration)), |
|
|
|
None => { |
|
|
|
None => { |
|
|
|
warn!( |
|
|
|
warn!( |
|
|
|
program_id = run.program.id, |
|
|
|
program_id = run.program.id, |
|
|
|
section_id = item.section_id, |
|
|
|
section_id = item.section_id, |
|
|
|
"trying to run program with nonexistant section" |
|
|
|
"trying to run program with nonexistant section" |
|
|
|
); |
|
|
|
); |
|
|
|
continue; |
|
|
|
None |
|
|
|
} |
|
|
|
} |
|
|
|
}; |
|
|
|
}) |
|
|
|
let handle = self |
|
|
|
.collect(); |
|
|
|
.section_runner |
|
|
|
if sequence.is_empty() { |
|
|
|
.queue_run(section, item.duration) |
|
|
|
warn!(program_id = run.program.id, "program has no valid sections"); |
|
|
|
.await |
|
|
|
run.state = RunState::Finished; |
|
|
|
.wrap_err("failed to queue section run")?; |
|
|
|
self.send_event(ProgramEvent::RunStart(run.program.clone())); |
|
|
|
|
|
|
|
self.send_event(ProgramEvent::RunFinish(run.program.clone())); |
|
|
|
|
|
|
|
return; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
run.sec_run_handles.reserve(sequence.len()); |
|
|
|
|
|
|
|
for (section, duration) in sequence { |
|
|
|
|
|
|
|
let handle = self.section_runner.do_queue_run(section, duration); |
|
|
|
run.sec_run_handles.push(handle); |
|
|
|
run.sec_run_handles.push(handle); |
|
|
|
} |
|
|
|
} |
|
|
|
run.state = RunState::Running; |
|
|
|
run.state = RunState::Running; |
|
|
|
self.send_event(ProgramEvent::RunStart(run.program.clone())); |
|
|
|
self.send_event(ProgramEvent::RunStart(run.program.clone())); |
|
|
|
if run.sec_run_handles.is_empty() { |
|
|
|
|
|
|
|
warn!(program_id = run.program.id, "program has no valid sections"); |
|
|
|
|
|
|
|
run.state = RunState::Finished; |
|
|
|
|
|
|
|
self.send_event(ProgramEvent::RunFinish(run.program.clone())); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
debug!(program_id = run.program.id, "started running program"); |
|
|
|
debug!(program_id = run.program.id, "started running program"); |
|
|
|
} |
|
|
|
} |
|
|
|
Ok(()) |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
async fn cancel_program_run(&mut self, run: &mut ProgRun) -> eyre::Result<()> { |
|
|
|
fn cancel_program_run(&mut self, run: &mut ProgRun) { |
|
|
|
for handle in run.sec_run_handles.drain(..) { |
|
|
|
for handle in run.sec_run_handles.drain(..) { |
|
|
|
self.section_runner |
|
|
|
self.section_runner.do_cancel_run(handle); |
|
|
|
.cancel_run(handle) |
|
|
|
|
|
|
|
.await |
|
|
|
|
|
|
|
.wrap_err("failed to cancel section run")?; |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
debug!(program_id = run.program.id, "program run is cancelled"); |
|
|
|
debug!(program_id = run.program.id, "program run is cancelled"); |
|
|
|
self.send_event(ProgramEvent::RunCancel(run.program.clone())); |
|
|
|
self.send_event(ProgramEvent::RunCancel(run.program.clone())); |
|
|
|
Ok(()) |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
async fn process_queue(&mut self, run_queue: &mut RunQueue) -> eyre::Result<()> { |
|
|
|
fn update_schedules(&mut self, ctx: &mut actix::Context<ProgramRunnerActor>) { |
|
|
|
while let Some(current_run) = run_queue.front_mut() { |
|
|
|
let mut scheduled_run_queue = DelayQueue::with_capacity(self.programs.len()); |
|
|
|
let run_finished = match current_run.state { |
|
|
|
|
|
|
|
RunState::Waiting => { |
|
|
|
|
|
|
|
self.start_program_run(current_run) |
|
|
|
|
|
|
|
.await |
|
|
|
|
|
|
|
.wrap_err("failed to start program run")?; |
|
|
|
|
|
|
|
false |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
RunState::Running => false, |
|
|
|
|
|
|
|
RunState::Finished => true, |
|
|
|
|
|
|
|
RunState::Cancelled => { |
|
|
|
|
|
|
|
self.cancel_program_run(current_run) |
|
|
|
|
|
|
|
.await |
|
|
|
|
|
|
|
.wrap_err("failed to cancel program run")?; |
|
|
|
|
|
|
|
true |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
}; |
|
|
|
|
|
|
|
if run_finished { |
|
|
|
|
|
|
|
run_queue.pop_front(); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
break; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
Ok(()) |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
fn update_programs(&mut self, new_programs: Programs) { |
|
|
|
|
|
|
|
self.programs = new_programs; |
|
|
|
|
|
|
|
self.scheduled_run_queue.clear(); |
|
|
|
|
|
|
|
for (_, prog) in &self.programs { |
|
|
|
for (_, prog) in &self.programs { |
|
|
|
if !prog.enabled { |
|
|
|
if !prog.enabled { |
|
|
|
continue; |
|
|
|
continue; |
|
|
@ -202,59 +146,217 @@ impl RunnerTask { |
|
|
|
}; |
|
|
|
}; |
|
|
|
let delay = (next_run - ref_time).to_std().unwrap(); |
|
|
|
let delay = (next_run - ref_time).to_std().unwrap(); |
|
|
|
trace!("will run program in {:?}", delay); |
|
|
|
trace!("will run program in {:?}", delay); |
|
|
|
self.scheduled_run_queue.insert(prog.clone(), delay); |
|
|
|
scheduled_run_queue.insert(prog.clone(), delay); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
let fut = actix::fut::wrap_stream(scheduled_run_queue) |
|
|
|
|
|
|
|
.map(|item, act: &mut ProgramRunnerActor, ctx| act.handle_scheduled_run(item, ctx)) |
|
|
|
|
|
|
|
.finish(); |
|
|
|
|
|
|
|
let handle = ctx.spawn(fut); |
|
|
|
|
|
|
|
if let Some(old_handle) = self.schedule_run_fut.replace(handle) { |
|
|
|
|
|
|
|
ctx.cancel_future(old_handle); |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
struct ProgramRunnerActor { |
|
|
|
|
|
|
|
inner: ProgramRunnerInner, |
|
|
|
|
|
|
|
run_queue: RunQueue, |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
fn handle_msg(&mut self, msg: Option<RunnerMsg>, run_queue: &mut RunQueue) { |
|
|
|
impl Actor for ProgramRunnerActor { |
|
|
|
let msg = msg.expect("ProgramRunner channel closed"); |
|
|
|
type Context = actix::Context<Self>; |
|
|
|
use RunnerMsg::*; |
|
|
|
|
|
|
|
trace!(msg = debug(&msg), "runner_task recv"); |
|
|
|
fn started(&mut self, ctx: &mut Self::Context) { |
|
|
|
match msg { |
|
|
|
trace!("subscribing to SectionRunner events"); |
|
|
|
Quit(quit_tx) => { |
|
|
|
let subscribe_fut = self.inner.section_runner.subscribe().into_actor(self).map( |
|
|
|
self.running = false; |
|
|
|
|section_events: Result<SectionEventRecv, SectionRunnerError>, |
|
|
|
self.quit_tx = Some(quit_tx); |
|
|
|
_act: &mut ProgramRunnerActor, |
|
|
|
|
|
|
|
ctx: &mut Self::Context| { |
|
|
|
|
|
|
|
match section_events { |
|
|
|
|
|
|
|
Ok(section_events) => { |
|
|
|
|
|
|
|
ctx.add_stream(section_events.into_stream()); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
Err(err) => warn!("failed to subscribe to SectionRunner events: {}", err), |
|
|
|
} |
|
|
|
} |
|
|
|
Subscribe(res_send) => { |
|
|
|
}, |
|
|
|
let event_recv = self.subscribe_event(); |
|
|
|
); |
|
|
|
// Ignore error if channel closed
|
|
|
|
ctx.wait(subscribe_fut); |
|
|
|
let _ = res_send.send(event_recv); |
|
|
|
trace!("program_runner starting"); |
|
|
|
} |
|
|
|
} |
|
|
|
UpdateSections(new_sections) => { |
|
|
|
|
|
|
|
self.sections = new_sections; |
|
|
|
fn stopped(&mut self, _ctx: &mut Self::Context) { |
|
|
|
|
|
|
|
trace!("program_runner stopped"); |
|
|
|
} |
|
|
|
} |
|
|
|
UpdatePrograms(new_programs) => { |
|
|
|
} |
|
|
|
self.update_programs(new_programs); |
|
|
|
|
|
|
|
|
|
|
|
impl StreamHandler<Result<SectionEvent, broadcast::RecvError>> for ProgramRunnerActor { |
|
|
|
|
|
|
|
fn handle( |
|
|
|
|
|
|
|
&mut self, |
|
|
|
|
|
|
|
item: Result<SectionEvent, broadcast::RecvError>, |
|
|
|
|
|
|
|
_ctx: &mut Self::Context, |
|
|
|
|
|
|
|
) { |
|
|
|
|
|
|
|
let sec_event = match item { |
|
|
|
|
|
|
|
Ok(e) => e, |
|
|
|
|
|
|
|
Err(err) => { |
|
|
|
|
|
|
|
warn!("failed to receive section event: {}", err); |
|
|
|
|
|
|
|
return; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
}; |
|
|
|
|
|
|
|
#[allow(clippy::single_match)] |
|
|
|
|
|
|
|
match sec_event { |
|
|
|
|
|
|
|
SectionEvent::RunFinish(finished_run, _) => { |
|
|
|
|
|
|
|
self.handle_finished_run(finished_run); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
_ => {} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#[derive(Message)] |
|
|
|
|
|
|
|
#[rtype(result = "()")] |
|
|
|
|
|
|
|
struct Quit; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
impl Handler<Quit> for ProgramRunnerActor { |
|
|
|
|
|
|
|
type Result = (); |
|
|
|
|
|
|
|
fn handle(&mut self, _msg: Quit, ctx: &mut Self::Context) -> Self::Result { |
|
|
|
|
|
|
|
ctx.stop(); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#[derive(Message)] |
|
|
|
|
|
|
|
#[rtype(result = "ProgramEventRecv")] |
|
|
|
|
|
|
|
struct Subscribe; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
impl Handler<Subscribe> for ProgramRunnerActor { |
|
|
|
|
|
|
|
type Result = MessageResult<Subscribe>; |
|
|
|
|
|
|
|
fn handle(&mut self, _msg: Subscribe, _ctx: &mut Self::Context) -> Self::Result { |
|
|
|
|
|
|
|
let event_recv = self.inner.subscribe_event(); |
|
|
|
|
|
|
|
MessageResult(event_recv) |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#[derive(Message)] |
|
|
|
|
|
|
|
#[rtype(result = "()")] |
|
|
|
|
|
|
|
struct UpdateSections(Sections); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
impl Handler<UpdateSections> for ProgramRunnerActor { |
|
|
|
|
|
|
|
type Result = (); |
|
|
|
|
|
|
|
fn handle(&mut self, msg: UpdateSections, _ctx: &mut Self::Context) -> Self::Result { |
|
|
|
|
|
|
|
trace!("updating sections"); |
|
|
|
|
|
|
|
let UpdateSections(new_sections) = msg; |
|
|
|
|
|
|
|
self.inner.sections = new_sections; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#[derive(Message)] |
|
|
|
|
|
|
|
#[rtype(result = "()")] |
|
|
|
|
|
|
|
struct UpdatePrograms(Programs); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
impl Handler<UpdatePrograms> for ProgramRunnerActor { |
|
|
|
|
|
|
|
type Result = (); |
|
|
|
|
|
|
|
fn handle(&mut self, msg: UpdatePrograms, ctx: &mut Self::Context) -> Self::Result { |
|
|
|
|
|
|
|
trace!("updating programs"); |
|
|
|
|
|
|
|
let UpdatePrograms(new_programs) = msg; |
|
|
|
|
|
|
|
self.inner.programs = new_programs; |
|
|
|
|
|
|
|
self.inner.update_schedules(ctx); |
|
|
|
} |
|
|
|
} |
|
|
|
RunProgramId(program_id) => { |
|
|
|
} |
|
|
|
let program = match self.programs.get(&program_id) { |
|
|
|
|
|
|
|
|
|
|
|
#[derive(Message)] |
|
|
|
|
|
|
|
#[rtype(result = "()")] |
|
|
|
|
|
|
|
struct RunProgramId(ProgramId); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
impl Handler<RunProgramId> for ProgramRunnerActor { |
|
|
|
|
|
|
|
type Result = (); |
|
|
|
|
|
|
|
fn handle(&mut self, msg: RunProgramId, ctx: &mut Self::Context) -> Self::Result { |
|
|
|
|
|
|
|
let RunProgramId(program_id) = msg; |
|
|
|
|
|
|
|
let program = match self.inner.programs.get(&program_id) { |
|
|
|
Some(program) => program.clone(), |
|
|
|
Some(program) => program.clone(), |
|
|
|
None => { |
|
|
|
None => { |
|
|
|
warn!(program_id, "trying to run non-existant program"); |
|
|
|
warn!(program_id, "trying to run non-existant program"); |
|
|
|
return; |
|
|
|
return; |
|
|
|
} |
|
|
|
} |
|
|
|
}; |
|
|
|
}; |
|
|
|
run_queue.push_back(ProgRun::new(program)); |
|
|
|
self.run_queue.push_back(ProgRun::new(program)); |
|
|
|
|
|
|
|
ctx.notify(Process); |
|
|
|
} |
|
|
|
} |
|
|
|
RunProgram(program) => { |
|
|
|
} |
|
|
|
run_queue.push_back(ProgRun::new(program)); |
|
|
|
|
|
|
|
|
|
|
|
#[derive(Message)] |
|
|
|
|
|
|
|
#[rtype(result = "()")] |
|
|
|
|
|
|
|
struct RunProgram(ProgramRef); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
impl Handler<RunProgram> for ProgramRunnerActor { |
|
|
|
|
|
|
|
type Result = (); |
|
|
|
|
|
|
|
fn handle(&mut self, msg: RunProgram, ctx: &mut Self::Context) -> Self::Result { |
|
|
|
|
|
|
|
let RunProgram(program) = msg; |
|
|
|
|
|
|
|
self.run_queue.push_back(ProgRun::new(program)); |
|
|
|
|
|
|
|
ctx.notify(Process); |
|
|
|
} |
|
|
|
} |
|
|
|
RunnerMsg::CancelProgram(program_id) => { |
|
|
|
} |
|
|
|
for run in run_queue { |
|
|
|
|
|
|
|
|
|
|
|
#[derive(Message)] |
|
|
|
|
|
|
|
#[rtype(result = "()")] |
|
|
|
|
|
|
|
struct CancelProgram(ProgramId); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
impl Handler<CancelProgram> for ProgramRunnerActor { |
|
|
|
|
|
|
|
type Result = (); |
|
|
|
|
|
|
|
fn handle(&mut self, msg: CancelProgram, ctx: &mut Self::Context) -> Self::Result { |
|
|
|
|
|
|
|
let CancelProgram(program_id) = msg; |
|
|
|
|
|
|
|
for run in self.run_queue.iter_mut() { |
|
|
|
if run.program.id == program_id { |
|
|
|
if run.program.id == program_id { |
|
|
|
run.state = RunState::Cancelled; |
|
|
|
run.state = RunState::Cancelled; |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
ctx.notify(Process); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#[derive(Message)] |
|
|
|
|
|
|
|
#[rtype(result = "()")] |
|
|
|
|
|
|
|
struct Process; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
impl Handler<Process> for ProgramRunnerActor { |
|
|
|
|
|
|
|
type Result = (); |
|
|
|
|
|
|
|
fn handle(&mut self, _msg: Process, _ctx: &mut Self::Context) -> Self::Result { |
|
|
|
|
|
|
|
while let Some(current_run) = self.run_queue.front_mut() { |
|
|
|
|
|
|
|
let run_finished = match current_run.state { |
|
|
|
|
|
|
|
RunState::Waiting => { |
|
|
|
|
|
|
|
self.inner.start_program_run(current_run); |
|
|
|
|
|
|
|
false |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
RunState::Running => false, |
|
|
|
|
|
|
|
RunState::Finished => true, |
|
|
|
|
|
|
|
RunState::Cancelled => { |
|
|
|
|
|
|
|
self.inner.cancel_program_run(current_run); |
|
|
|
|
|
|
|
true |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
}; |
|
|
|
|
|
|
|
if run_finished { |
|
|
|
|
|
|
|
self.run_queue.pop_front(); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
break; |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
impl ProgramRunnerActor { |
|
|
|
|
|
|
|
fn new(section_runner: SectionRunner) -> Self { |
|
|
|
|
|
|
|
Self { |
|
|
|
|
|
|
|
inner: ProgramRunnerInner { |
|
|
|
|
|
|
|
section_runner, |
|
|
|
|
|
|
|
sections: Sections::new(), |
|
|
|
|
|
|
|
programs: Programs::new(), |
|
|
|
|
|
|
|
event_send: None, |
|
|
|
|
|
|
|
schedule_run_fut: None, |
|
|
|
|
|
|
|
}, |
|
|
|
|
|
|
|
run_queue: RunQueue::new(), |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
fn handle_finished_run( |
|
|
|
fn handle_finished_run(&mut self, finished_run: SectionRunHandle) -> Option<()> { |
|
|
|
&mut self, |
|
|
|
let current_run = self.run_queue.front_mut()?; |
|
|
|
finished_run: SectionRunHandle, |
|
|
|
|
|
|
|
run_queue: &mut RunQueue, |
|
|
|
|
|
|
|
) -> Option<()> { |
|
|
|
|
|
|
|
let current_run = run_queue.front_mut()?; |
|
|
|
|
|
|
|
let last_run_handle = current_run.sec_run_handles.last()?; |
|
|
|
let last_run_handle = current_run.sec_run_handles.last()?; |
|
|
|
if finished_run == *last_run_handle { |
|
|
|
if finished_run == *last_run_handle { |
|
|
|
current_run.state = RunState::Finished; |
|
|
|
current_run.state = RunState::Finished; |
|
|
@ -262,73 +364,27 @@ impl RunnerTask { |
|
|
|
program_id = current_run.program.id, |
|
|
|
program_id = current_run.program.id, |
|
|
|
"finished running program" |
|
|
|
"finished running program" |
|
|
|
); |
|
|
|
); |
|
|
|
self.send_event(ProgramEvent::RunFinish(current_run.program.clone())); |
|
|
|
self.inner |
|
|
|
|
|
|
|
.send_event(ProgramEvent::RunFinish(current_run.program.clone())); |
|
|
|
} |
|
|
|
} |
|
|
|
Some(()) |
|
|
|
Some(()) |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
fn handle_sec_event( |
|
|
|
fn handle_scheduled_run( |
|
|
|
&mut self, |
|
|
|
|
|
|
|
sec_event: Result<SectionEvent, broadcast::RecvError>, |
|
|
|
|
|
|
|
run_queue: &mut RunQueue, |
|
|
|
|
|
|
|
) -> eyre::Result<()> { |
|
|
|
|
|
|
|
let sec_event = sec_event.wrap_err("failed to receive section event")?; |
|
|
|
|
|
|
|
#[allow(clippy::single_match)] |
|
|
|
|
|
|
|
match sec_event { |
|
|
|
|
|
|
|
SectionEvent::RunFinish(finished_run, _) => { |
|
|
|
|
|
|
|
self.handle_finished_run(finished_run, run_queue); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
_ => {} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
Ok(()) |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
async fn handle_scheduled_run( |
|
|
|
|
|
|
|
&mut self, |
|
|
|
&mut self, |
|
|
|
item: Result<delay_queue::Expired<ProgramRef>, tokio::time::Error>, |
|
|
|
item: Result<delay_queue::Expired<ProgramRef>, tokio::time::Error>, |
|
|
|
run_queue: &mut RunQueue, |
|
|
|
ctx: &mut <ProgramRunnerActor as Actor>::Context, |
|
|
|
) -> eyre::Result<()> { |
|
|
|
) { |
|
|
|
let item = item.wrap_err("tokio time error")?; |
|
|
|
let program = match item { |
|
|
|
run_queue.push_back(ProgRun::new(item.into_inner())); |
|
|
|
Ok(expired) => expired.into_inner(), |
|
|
|
Ok(()) |
|
|
|
Err(err) => { |
|
|
|
|
|
|
|
error!("tokio time error: {}", err); |
|
|
|
|
|
|
|
return; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
async fn run_impl(&mut self) -> eyre::Result<()> { |
|
|
|
|
|
|
|
let mut sec_events = self |
|
|
|
|
|
|
|
.section_runner |
|
|
|
|
|
|
|
.subscribe() |
|
|
|
|
|
|
|
.await |
|
|
|
|
|
|
|
.wrap_err("could not subscribe to SectionRunner events")?; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
let mut run_queue: RunQueue = VecDeque::new(); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
while self.running { |
|
|
|
|
|
|
|
self.process_queue(&mut run_queue) |
|
|
|
|
|
|
|
.await |
|
|
|
|
|
|
|
.wrap_err("error during queue processing")?; |
|
|
|
|
|
|
|
tokio::select! { |
|
|
|
|
|
|
|
msg = self.msg_recv.recv() => self.handle_msg(msg, &mut run_queue), |
|
|
|
|
|
|
|
sec_event = sec_events.recv() => self.handle_sec_event(sec_event, &mut run_queue)?, |
|
|
|
|
|
|
|
Some(scheduled_run) = self.scheduled_run_queue.next() => { |
|
|
|
|
|
|
|
self.handle_scheduled_run(scheduled_run, &mut run_queue).await?; |
|
|
|
|
|
|
|
}, |
|
|
|
|
|
|
|
}; |
|
|
|
}; |
|
|
|
} |
|
|
|
trace!(program_id = program.id, "schedule expired"); |
|
|
|
|
|
|
|
self.run_queue.push_back(ProgRun::new(program)); |
|
|
|
if let Some(quit_tx) = self.quit_tx.take() { |
|
|
|
ctx.notify(Process); |
|
|
|
let _ = quit_tx.send(()); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Ok(()) |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
async fn run(mut self) { |
|
|
|
|
|
|
|
let span = trace_span!("program_runner task"); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
self.run_impl() |
|
|
|
|
|
|
|
.instrument(span) |
|
|
|
|
|
|
|
.await |
|
|
|
|
|
|
|
.expect("error in ProgramRunner task"); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
@ -350,65 +406,67 @@ impl From<oneshot::error::RecvError> for ChannelClosed { |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
#[derive(Clone, Debug)] |
|
|
|
impl From<actix::MailboxError> for ChannelClosed { |
|
|
|
|
|
|
|
fn from(_: actix::MailboxError) -> Self { |
|
|
|
|
|
|
|
// TODO:
|
|
|
|
|
|
|
|
Self |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#[derive(Clone)] |
|
|
|
pub struct ProgramRunner { |
|
|
|
pub struct ProgramRunner { |
|
|
|
msg_send: mpsc::Sender<RunnerMsg>, |
|
|
|
addr: Addr<ProgramRunnerActor>, |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
#[allow(dead_code)] |
|
|
|
#[allow(dead_code)] |
|
|
|
impl ProgramRunner { |
|
|
|
impl ProgramRunner { |
|
|
|
pub fn new(section_runner: SectionRunner) -> Self { |
|
|
|
pub fn new(section_runner: SectionRunner) -> Self { |
|
|
|
let (msg_send, msg_recv) = mpsc::channel(8); |
|
|
|
let addr = ProgramRunnerActor::new(section_runner).start(); |
|
|
|
spawn(RunnerTask::new(section_runner, msg_recv).run()); |
|
|
|
Self { addr } |
|
|
|
Self { msg_send } |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
pub async fn quit(&mut self) -> Result<()> { |
|
|
|
pub async fn quit(&mut self) -> Result<()> { |
|
|
|
let (quit_tx, quit_rx) = oneshot::channel(); |
|
|
|
self.addr.send(Quit).await?; |
|
|
|
self.msg_send.send(RunnerMsg::Quit(quit_tx)).await?; |
|
|
|
|
|
|
|
quit_rx.await?; |
|
|
|
|
|
|
|
Ok(()) |
|
|
|
Ok(()) |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
pub async fn update_sections(&mut self, new_sections: Sections) -> Result<()> { |
|
|
|
pub async fn update_sections(&mut self, new_sections: Sections) -> Result<()> { |
|
|
|
self.msg_send |
|
|
|
self.addr |
|
|
|
.send(RunnerMsg::UpdateSections(new_sections)) |
|
|
|
.send(UpdateSections(new_sections)) |
|
|
|
.await |
|
|
|
.await |
|
|
|
.map_err(From::from) |
|
|
|
.map_err(From::from) |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
pub async fn update_programs(&mut self, new_programs: Programs) -> Result<()> { |
|
|
|
pub async fn update_programs(&mut self, new_programs: Programs) -> Result<()> { |
|
|
|
self.msg_send |
|
|
|
self.addr |
|
|
|
.send(RunnerMsg::UpdatePrograms(new_programs)) |
|
|
|
.send(UpdatePrograms(new_programs)) |
|
|
|
.await |
|
|
|
.await |
|
|
|
.map_err(From::from) |
|
|
|
.map_err(From::from) |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
pub async fn run_program_id(&mut self, program_id: ProgramId) -> Result<()> { |
|
|
|
pub async fn run_program_id(&mut self, program_id: ProgramId) -> Result<()> { |
|
|
|
self.msg_send |
|
|
|
self.addr |
|
|
|
.send(RunnerMsg::RunProgramId(program_id)) |
|
|
|
.send(RunProgramId(program_id)) |
|
|
|
.await |
|
|
|
.await |
|
|
|
.map_err(From::from) |
|
|
|
.map_err(From::from) |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
pub async fn run_program(&mut self, program: ProgramRef) -> Result<()> { |
|
|
|
pub async fn run_program(&mut self, program: ProgramRef) -> Result<()> { |
|
|
|
self.msg_send |
|
|
|
self.addr |
|
|
|
.send(RunnerMsg::RunProgram(program)) |
|
|
|
.send(RunProgram(program)) |
|
|
|
.await |
|
|
|
.await |
|
|
|
.map_err(From::from) |
|
|
|
.map_err(From::from) |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
pub async fn cancel_program(&mut self, program_id: ProgramId) -> Result<()> { |
|
|
|
pub async fn cancel_program(&mut self, program_id: ProgramId) -> Result<()> { |
|
|
|
self.msg_send |
|
|
|
self.addr |
|
|
|
.send(RunnerMsg::CancelProgram(program_id)) |
|
|
|
.send(CancelProgram(program_id)) |
|
|
|
.await |
|
|
|
.await |
|
|
|
.map_err(From::from) |
|
|
|
.map_err(From::from) |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
pub async fn subscribe(&mut self) -> Result<ProgramEventRecv> { |
|
|
|
pub async fn subscribe(&mut self) -> Result<ProgramEventRecv> { |
|
|
|
let (res_send, res_recv) = oneshot::channel(); |
|
|
|
let event_recv = self.addr.send(Subscribe).await?; |
|
|
|
self.msg_send.send(RunnerMsg::Subscribe(res_send)).await?; |
|
|
|
|
|
|
|
let event_recv = res_recv.await?; |
|
|
|
|
|
|
|
Ok(event_recv) |
|
|
|
Ok(event_recv) |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
@ -420,7 +478,7 @@ mod test { |
|
|
|
use crate::{ |
|
|
|
use crate::{ |
|
|
|
model::{Program, ProgramItem, Section}, |
|
|
|
model::{Program, ProgramItem, Section}, |
|
|
|
schedule::{every_day, DateTimeBound, Schedule}, |
|
|
|
schedule::{every_day, DateTimeBound, Schedule}, |
|
|
|
trace_listeners::{EventListener, Filters, SpanFilters, SpanListener}, |
|
|
|
trace_listeners::{EventListener, Filters}, |
|
|
|
}; |
|
|
|
}; |
|
|
|
use assert_matches::assert_matches; |
|
|
|
use assert_matches::assert_matches; |
|
|
|
use im::ordmap; |
|
|
|
use im::ordmap; |
|
|
@ -433,17 +491,9 @@ mod test { |
|
|
|
let quit_msg = EventListener::new( |
|
|
|
let quit_msg = EventListener::new( |
|
|
|
Filters::new() |
|
|
|
Filters::new() |
|
|
|
.target("sprinklers_rs::program_runner") |
|
|
|
.target("sprinklers_rs::program_runner") |
|
|
|
.message("runner_task recv") |
|
|
|
.message("program_runner stopped"), |
|
|
|
.field_value("msg", "Quit"), |
|
|
|
|
|
|
|
); |
|
|
|
|
|
|
|
let task_span = SpanListener::new( |
|
|
|
|
|
|
|
SpanFilters::new() |
|
|
|
|
|
|
|
.target("sprinklers_rs::program_runner") |
|
|
|
|
|
|
|
.name("program_runner task"), |
|
|
|
|
|
|
|
); |
|
|
|
); |
|
|
|
let subscriber = tracing_subscriber::registry() |
|
|
|
let subscriber = tracing_subscriber::registry().with(quit_msg.clone()); |
|
|
|
.with(quit_msg.clone()) |
|
|
|
|
|
|
|
.with(task_span.clone()); |
|
|
|
|
|
|
|
let _sub = tracing::subscriber::set_default(subscriber); |
|
|
|
let _sub = tracing::subscriber::set_default(subscriber); |
|
|
|
|
|
|
|
|
|
|
|
let interface = MockSectionInterface::new(6); |
|
|
|
let interface = MockSectionInterface::new(6); |
|
|
@ -455,7 +505,6 @@ mod test { |
|
|
|
yield_now().await; |
|
|
|
yield_now().await; |
|
|
|
|
|
|
|
|
|
|
|
assert_eq!(quit_msg.get_count(), 1); |
|
|
|
assert_eq!(quit_msg.get_count(), 1); |
|
|
|
assert!(task_span.get_exit_count() > 1); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
fn make_sections_and_runner() -> (Sections, SectionRunner, Arc<MockSectionInterface>) { |
|
|
|
fn make_sections_and_runner() -> (Sections, SectionRunner, Arc<MockSectionInterface>) { |
|
|
@ -530,24 +579,12 @@ mod test { |
|
|
|
assert_eq!(interface.get_section_state(0), true); |
|
|
|
assert_eq!(interface.get_section_state(0), true); |
|
|
|
|
|
|
|
|
|
|
|
tokio::time::pause(); |
|
|
|
tokio::time::pause(); |
|
|
|
assert_matches!( |
|
|
|
assert_matches!(sec_events.recv().await, Ok(SectionEvent::RunFinish(_, _))); |
|
|
|
sec_events.recv().await, |
|
|
|
assert_matches!(sec_events.recv().await, Ok(SectionEvent::RunStart(_, _))); |
|
|
|
Ok(SectionEvent::RunFinish(_, _)) |
|
|
|
|
|
|
|
); |
|
|
|
|
|
|
|
assert_matches!( |
|
|
|
|
|
|
|
sec_events.recv().await, |
|
|
|
|
|
|
|
Ok(SectionEvent::RunStart(_, _)) |
|
|
|
|
|
|
|
); |
|
|
|
|
|
|
|
assert_eq!(interface.get_section_state(0), false); |
|
|
|
assert_eq!(interface.get_section_state(0), false); |
|
|
|
assert_eq!(interface.get_section_state(1), true); |
|
|
|
assert_eq!(interface.get_section_state(1), true); |
|
|
|
assert_matches!( |
|
|
|
assert_matches!(sec_events.recv().await, Ok(SectionEvent::RunFinish(_, _))); |
|
|
|
sec_events.recv().await, |
|
|
|
assert_matches!(prog_events.recv().await, Ok(ProgramEvent::RunFinish(_))); |
|
|
|
Ok(SectionEvent::RunFinish(_, _)) |
|
|
|
|
|
|
|
); |
|
|
|
|
|
|
|
assert_matches!( |
|
|
|
|
|
|
|
prog_events.recv().await, |
|
|
|
|
|
|
|
Ok(ProgramEvent::RunFinish(_)) |
|
|
|
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
runner.quit().await.unwrap(); |
|
|
|
runner.quit().await.unwrap(); |
|
|
|
sec_runner.quit().await.unwrap(); |
|
|
|
sec_runner.quit().await.unwrap(); |
|
|
@ -581,12 +618,12 @@ mod test { |
|
|
|
// Should immediately start and finish running program
|
|
|
|
// Should immediately start and finish running program
|
|
|
|
// due to nonexistant section
|
|
|
|
// due to nonexistant section
|
|
|
|
assert_matches!( |
|
|
|
assert_matches!( |
|
|
|
prog_events.try_recv(), |
|
|
|
prog_events.recv().await, |
|
|
|
Ok(ProgramEvent::RunStart(prog)) |
|
|
|
Ok(ProgramEvent::RunStart(prog)) |
|
|
|
if prog.id == 1 |
|
|
|
if prog.id == 1 |
|
|
|
); |
|
|
|
); |
|
|
|
assert_matches!( |
|
|
|
assert_matches!( |
|
|
|
prog_events.try_recv(), |
|
|
|
prog_events.recv().await, |
|
|
|
Ok(ProgramEvent::RunFinish(prog)) |
|
|
|
Ok(ProgramEvent::RunFinish(prog)) |
|
|
|
if prog.id == 1 |
|
|
|
if prog.id == 1 |
|
|
|
); |
|
|
|
); |
|
|
@ -734,10 +771,7 @@ mod test { |
|
|
|
Ok(ProgramEvent::RunCancel(prog)) |
|
|
|
Ok(ProgramEvent::RunCancel(prog)) |
|
|
|
if prog.id == 1 |
|
|
|
if prog.id == 1 |
|
|
|
); |
|
|
|
); |
|
|
|
assert_matches!( |
|
|
|
assert_matches!(sec_events.recv().await, Ok(SectionEvent::RunCancel(_, _))); |
|
|
|
sec_events.recv().await, |
|
|
|
|
|
|
|
Ok(SectionEvent::RunCancel(_, _)) |
|
|
|
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
runner.quit().await.unwrap(); |
|
|
|
runner.quit().await.unwrap(); |
|
|
|
sec_runner.quit().await.unwrap(); |
|
|
|
sec_runner.quit().await.unwrap(); |
|
|
@ -745,7 +779,6 @@ mod test { |
|
|
|
|
|
|
|
|
|
|
|
#[actix_rt::test] |
|
|
|
#[actix_rt::test] |
|
|
|
async fn test_scheduled_run() { |
|
|
|
async fn test_scheduled_run() { |
|
|
|
tracing_subscriber::fmt().init(); |
|
|
|
|
|
|
|
let (sections, mut sec_runner, _) = make_sections_and_runner(); |
|
|
|
let (sections, mut sec_runner, _) = make_sections_and_runner(); |
|
|
|
let mut runner = ProgramRunner::new(sec_runner.clone()); |
|
|
|
let mut runner = ProgramRunner::new(sec_runner.clone()); |
|
|
|
let mut prog_events = runner.subscribe().await.unwrap(); |
|
|
|
let mut prog_events = runner.subscribe().await.unwrap(); |
|
|
|