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,
};

/// This is a graceful stop for daemons.
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
            }));
        }
    }
}

/// This is a stop message for actors.
#[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();
    }
}

/// This is a system terminator. This will stop all Arc<SystemTerminator> released than is having actors.
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();
            }
        }
    }
}