1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
use crate::delayer::{Delayer, Task};
use actix::prelude::*;
use actix_rt::signal::unix::{signal, SignalKind};
use futures_util::stream::once;
use std::{
sync::{mpsc::Sender, Arc},
time::Duration,
};
pub struct GracefulStop {
stop_request_recipients: Vec<Recipient<StopRequest>>,
system_terminator: Arc<SystemTerminator>,
}
impl GracefulStop {
pub fn new() -> GracefulStop {
GracefulStop {
stop_request_recipients: Vec::new(),
system_terminator: Arc::new(SystemTerminator { sender: None }),
}
}
pub fn new_with_sender(sender: Sender<()>) -> GracefulStop {
GracefulStop {
stop_request_recipients: Vec::new(),
system_terminator: Arc::new(SystemTerminator {
sender: Some(sender),
}),
}
}
pub fn subscribe(mut self, recipient: Recipient<StopRequest>) -> Self {
self.stop_request_recipients.push(recipient);
self
}
pub fn clone_system_terminator(&self) -> Arc<SystemTerminator> {
Arc::clone(&self.system_terminator)
}
pub fn subscribe_ref(&mut self, recipient: Recipient<StopRequest>) {
self.stop_request_recipients.push(recipient);
}
pub fn start_with_delayers(receipts: Vec<(Recipient<Task>, Duration)>) -> Addr<Self> {
let mut graceful_stop = GracefulStop::new();
for receipt in receipts {
let delayer = Delayer::new(
receipt.0,
graceful_stop.clone_system_terminator(),
receipt.1,
)
.start();
graceful_stop.subscribe_ref(delayer.recipient());
}
graceful_stop.start()
}
}
impl Actor for GracefulStop {
type Context = Context<Self>;
fn started(&mut self, ctx: &mut Self::Context) {
let signals = vec![
Box::new(SignalKind::hangup()),
Box::new(SignalKind::interrupt()),
Box::new(SignalKind::quit()),
Box::new(SignalKind::terminate()),
];
for signal_kind in signals.into_iter() {
let mut s = signal(*signal_kind).unwrap();
ctx.add_message_stream(once(async move {
s.recv().await;
StopEvent
}));
}
}
}
#[derive(Message)]
#[rtype(result = "()")]
pub struct StopRequest;
#[derive(Message)]
#[rtype(result = "()")]
pub struct StopEvent;
impl Handler<StopEvent> for GracefulStop {
type Result = ();
fn handle(&mut self, _signal_event: StopEvent, ctx: &mut Self::Context) {
for recipient in self.stop_request_recipients.drain(..) {
let _ = recipient.do_send(StopRequest);
}
ctx.stop();
}
}
pub struct SystemTerminator {
sender: Option<Sender<()>>,
}
impl Drop for SystemTerminator {
fn drop(&mut self) {
match self.sender {
Some(ref sender) => {
sender.send(()).unwrap();
}
None => {
actix::System::current().stop();
}
}
}
}