refactor stop logic

This commit is contained in:
dignifiedquire
2020-03-21 19:05:50 +01:00
parent 4d6a9e4f14
commit d798356c19
4 changed files with 51 additions and 30 deletions

View File

@@ -268,13 +268,15 @@ async fn start(args: Vec<String>) -> Result<(), failure::Error> {
let context = Context::new("CLI".into(), Path::new(&args[1]).to_path_buf()).await?; let context = Context::new("CLI".into(), Path::new(&args[1]).to_path_buf()).await?;
let ctx = context.clone(); let ctx = context.clone();
std::thread::spawn(move || loop { async_std::task::spawn(async move {
loop {
if ctx.has_next_event() { if ctx.has_next_event() {
if let Ok(event) = ctx.get_next_event() { if let Ok(event) = ctx.get_next_event() {
receive_event(event); receive_event(event);
} }
} else { } else {
std::thread::sleep(std::time::Duration::from_millis(50)); async_std::task::sleep(std::time::Duration::from_millis(50)).await;
}
} }
}); });
@@ -316,6 +318,7 @@ async fn start(args: Vec<String>) -> Result<(), failure::Error> {
} }
Err(ReadlineError::Interrupted) | Err(ReadlineError::Eof) => { Err(ReadlineError::Interrupted) | Err(ReadlineError::Eof) => {
println!("Exiting..."); println!("Exiting...");
context.stop().await;
break; break;
} }
Err(err) => { Err(err) => {
@@ -324,11 +327,10 @@ async fn start(args: Vec<String>) -> Result<(), failure::Error> {
} }
} }
} }
rl.save_history(".dc-history.txt")?; rl.save_history(".dc-history.txt")?;
println!("history saved"); println!("history saved");
context.stop().await;
Ok(()) Ok(())
} }

View File

@@ -39,13 +39,15 @@ async fn main() {
println!("info: {:#?}", info); println!("info: {:#?}", info);
let ctx1 = ctx.clone(); let ctx1 = ctx.clone();
std::thread::spawn(move || loop { async_std::task::spawn(async move {
loop {
if ctx1.has_next_event() { if ctx1.has_next_event() {
if let Ok(event) = ctx1.get_next_event() { if let Ok(event) = ctx1.get_next_event() {
cb(event); cb(event);
} }
} else { } else {
std::thread::sleep(time::Duration::from_millis(50)); async_std::task::sleep(time::Duration::from_millis(50));
}
} }
}); });

View File

@@ -128,8 +128,8 @@ impl Context {
} }
pub async fn run(&self) { pub async fn run(&self) {
if self.inner.scheduler.read().await.is_running() { if self.is_running().await {
panic!("Already running"); return;
} }
let l = &mut *self.inner.scheduler.write().await; let l = &mut *self.inner.scheduler.write().await;
@@ -137,7 +137,7 @@ impl Context {
} }
pub async fn is_running(&self) -> bool { pub async fn is_running(&self) -> bool {
self.inner.scheduler.read().await.is_running() self.inner.is_running().await
} }
pub async fn stop(&self) { pub async fn stop(&self) {
@@ -480,19 +480,21 @@ impl Context {
} }
impl InnerContext { impl InnerContext {
async fn stop(&self) { async fn is_running(&self) -> bool {
if self.scheduler.read().await.is_running() { self.scheduler.read().await.is_running()
self.scheduler.write().await.stop().await;
}
}
} }
impl Drop for InnerContext { async fn stop(&self) {
fn drop(&mut self) { if self.is_running().await {
async_std::task::block_on(async move { let token = {
self.stop().await; let lock = &*self.scheduler.read().await;
self.sql.close().await; lock.pre_stop().await
}); };
{
let lock = &mut *self.scheduler.write().await;
lock.stop(token).await;
}
}
} }
} }

View File

@@ -9,6 +9,8 @@ use crate::imap::Imap;
use crate::job::{self, Thread}; use crate::job::{self, Thread};
use crate::smtp::Smtp; use crate::smtp::Smtp;
pub(crate) struct StopToken;
/// Job and connection scheduler. /// Job and connection scheduler.
#[derive(Debug)] #[derive(Debug)]
pub(crate) enum Scheduler { pub(crate) enum Scheduler {
@@ -287,8 +289,8 @@ impl Scheduler {
} }
} }
/// Halt the scheduler, panics if it is already stopped. /// Halts the scheduler, must be called first, and then `stop`.
pub async fn stop(&mut self) { pub(crate) async fn pre_stop(&self) -> StopToken {
match self { match self {
Scheduler::Stopped => { Scheduler::Stopped => {
panic!("WARN: already stopped"); panic!("WARN: already stopped");
@@ -306,6 +308,19 @@ impl Scheduler {
.join(sentbox.stop()) .join(sentbox.stop())
.join(smtp.stop()) .join(smtp.stop())
.await; .await;
StopToken
}
}
}
/// Halt the scheduler, must only be called after pre_stop.
pub(crate) async fn stop(&mut self, _t: StopToken) {
match self {
Scheduler::Stopped => {
panic!("WARN: already stopped");
}
Scheduler::Running { .. } => {
*self = Scheduler::Stopped; *self = Scheduler::Stopped;
} }
} }