aboutsummaryrefslogtreecommitdiff
path: root/src/main.rs
blob: 44bd6514dc19096ad64575ca64b5d23f571d094c (plain) (blame)
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 std::io::{self, Read, Write};
use std::net::{SocketAddr, TcpStream};
use std::time::Duration;

const MAGIC: [u8; 4] = [0x32, 0x7f, 0xfe, 0x4c];
const RESP_OK: [u8; 5] = [0x32, 0x7f, 0xfe, 0x4c, 0x00];
const RESP_NORMAL: [u8; 5] = [0x32, 0x7f, 0xfe, 0x4c, 0x01];
const PING_INTERVAL: Duration = Duration::from_secs(12);
const TCP_TIMEOUT: Duration = Duration::from_secs(8);
const MAX_RETRY: i32 = 30;

const EXIT_SUCCESS: i32 = 0;
const EXIT_USAGE: i32 = 1;
const EXIT_PING: i32 = 2;
const EXIT_BUG: i32 = 3;
const EXIT_IO: i32 = 4;

fn main() {
    let mut args = std::env::args().skip(1);

    let addr = match args.next() {
        Some(addr) => addr,
        None => {
            eprintln!("Usage: rsdsl_healthping host:port");
            std::process::exit(EXIT_USAGE);
        }
    };
    let addr: SocketAddr = match addr.parse() {
        Ok(addr) => addr,
        Err(e) => {
            eprintln!("invalid address: {}", e);
            std::process::exit(EXIT_USAGE);
        }
    };

    let mut lasterr = None;

    print!("Pinging");
    for _ in 0..MAX_RETRY {
        print!(".");
        match io::stdout().flush() {
            Ok(_) => {}
            Err(e) => {
                eprintln!("flush stdout: {}", e);

                std::thread::sleep(PING_INTERVAL);
                std::process::exit(EXIT_IO);
            }
        }

        let mut conn = match TcpStream::connect_timeout(&addr, TCP_TIMEOUT) {
            Ok(conn) => conn,
            Err(e) => {
                lasterr = Some(e);

                std::thread::sleep(PING_INTERVAL);
                continue;
            }
        };

        match conn.write_all(&MAGIC) {
            Ok(_) => {}
            Err(e) => {
                lasterr = Some(e);

                std::thread::sleep(PING_INTERVAL);
                continue;
            }
        }

        let mut buf = [0; 5];
        match conn.read_exact(&mut buf) {
            Ok(_) => {}
            Err(e) => {
                lasterr = Some(e);

                std::thread::sleep(PING_INTERVAL);
                continue;
            }
        }

        match buf {
            RESP_OK => success(),
            RESP_NORMAL => success_unexpected(),
            _ => eprintln!("got invalid response {:?}", buf),
        }

        std::thread::sleep(PING_INTERVAL);
    }

    println!();

    match lasterr {
        Some(e) => {
            eprintln!("{}", e);
            std::process::exit(EXIT_PING)
        }
        None => {
            eprintln!("max retries exceeded without error");
            std::process::exit(EXIT_BUG)
        }
    }
}

fn success() {
    println!();
    println!("Success");

    std::process::exit(EXIT_SUCCESS);
}

fn success_unexpected() {
    println!();
    println!("Host is not waiting for healthcheck ping");

    std::process::exit(EXIT_SUCCESS);
}