]> git.proxmox.com Git - proxmox-backup.git/blame - src/bin/proxmox-backup-api.rs
tools: daemon: rename some structs
[proxmox-backup.git] / src / bin / proxmox-backup-api.rs
CommitLineData
fe0e04c6 1extern crate proxmox_backup;
845901f4 2
d01e2420 3//use proxmox_backup::tools;
dc9a007b
DM
4use proxmox_backup::api_schema::router::*;
5use proxmox_backup::api_schema::config::*;
fe0e04c6 6use proxmox_backup::server::rest::*;
e4311382 7use proxmox_backup::tools::daemon::Reloader;
6c30068e 8use proxmox_backup::auth_helpers::*;
a8f268af 9use proxmox_backup::config;
886e5ce8 10
aa5a4060 11use failure::*;
16b48b81
DM
12use lazy_static::lazy_static;
13
9bc17e8d 14use futures::future::Future;
5e7bc50a 15use tokio::prelude::*;
b82472c0 16
9bc17e8d 17use hyper;
886e5ce8 18
5e7bc50a
WB
19static mut QUIT_MAIN: bool = false;
20
d8d978eb 21fn main() {
d8d978eb 22
aa5a4060
DM
23 if let Err(err) = run() {
24 eprintln!("Error: {}", err);
25 std::process::exit(-1);
26 }
27}
28
29fn run() -> Result<(), Error> {
5e7bc50a 30 // This manages data for reloads:
e4311382 31 let mut reloader = Reloader::new();
aa5a4060 32
d96d8273
DM
33 if let Err(err) = syslog::init(
34 syslog::Facility::LOG_DAEMON,
35 log::LevelFilter::Info,
36 Some("proxmox-backup-api")) {
aa5a4060 37 bail!("unable to inititialize syslog - {}", err);
a8f268af
DM
38 }
39
40 config::create_configdir()?;
d96d8273 41
39a90ca6 42 if let Err(err) = generate_auth_key() {
aa5a4060 43 bail!("unable to generate auth key - {}", err);
8d04280b 44 }
d01e2420 45 let _ = private_auth_key(); // load with lazy_static
8d04280b 46
39a90ca6 47 if let Err(err) = generate_csrf_key() {
aa5a4060 48 bail!("unable to generate csrf key - {}", err);
39a90ca6 49 }
d01e2420 50 let _ = csrf_secret(); // load with lazy_static
39a90ca6 51
a9696f7b 52 lazy_static!{
576e3bf2 53 static ref ROUTER: Router = proxmox_backup::api2::router();
a9696f7b 54 }
324a5bd0 55
02c7a755 56 let config = ApiConfig::new(
51ebd079 57 env!("PROXMOX_JSDIR"), &ROUTER, RpcEnvironmentType::PRIVILEGED);
324a5bd0 58
9bc17e8d 59 let rest_server = RestServer::new(config);
886e5ce8 60
5e7bc50a
WB
61 // http server future:
62
e4311382 63 let listener: tokio::net::TcpListener = reloader.restore(
5e7bc50a
WB
64 "PROXMOX_BACKUP_LISTEN_FD",
65 || {
66 let addr = ([127,0,0,1], 82).into();
67 Ok(tokio::net::TcpListener::bind(&addr)?)
68 },
69 )?;
70
71 let mut http_server = hyper::Server::builder(listener.incoming())
9bc17e8d 72 .serve(rest_server)
886e5ce8
DM
73 .map_err(|e| eprintln!("server error: {}", e));
74
5e7bc50a
WB
75 // signalfd future:
76
77 let signal_handler =
78 proxmox_backup::tools::daemon::default_signalfd_stream(
e4311382 79 reloader,
5e7bc50a
WB
80 || {
81 unsafe { QUIT_MAIN = true; }
82 Ok(())
83 },
84 )?
85 .map(|si| {
86 // debugging...
87 eprintln!("received signal: {}", si.ssi_signo);
88 })
89 .map_err(|e| {
90 eprintln!("error from signalfd: {}, shutting down...", e);
91 unsafe {
92 QUIT_MAIN = true;
93 }
94 });
95
96
97 // Combined future for signalfd & http server, we want to quit as soon as either of them ends.
98 // Neither of them is supposed to end unless some weird error happens, so just bail out if is
99 // the case...
100 let mut signal_handler = signal_handler.into_future();
101 let main = futures::future::poll_fn(move || {
102 // Helper for some diagnostic error messages:
103 fn poll_helper<S: Future>(stream: &mut S, name: &'static str) -> bool {
104 match stream.poll() {
105 Ok(Async::Ready(_)) => {
106 eprintln!("{} ended, shutting down", name);
107 true
108 }
109 Err(_) => {
110 eprintln!("{} error, shutting down", name);
111 true
112 },
113 _ => false,
114 }
115 }
116 if poll_helper(&mut http_server, "http server") ||
117 poll_helper(&mut signal_handler, "signalfd handler")
118 {
119 return Ok(Async::Ready(()));
120 }
121
122 if unsafe { QUIT_MAIN } {
123 eprintln!("shutdown requested");
124 Ok(Async::Ready(()))
125 } else {
126 Ok(Async::NotReady)
127 }
128 });
129
130 hyper::rt::run(main);
aa5a4060
DM
131
132 Ok(())
d8d978eb 133}