2021-09-30 11:49:29 +00:00
|
|
|
//! # Proxmox REST server
|
|
|
|
//!
|
|
|
|
//! This module provides convenient building blocks to implement a
|
|
|
|
//! REST server.
|
|
|
|
//!
|
|
|
|
//! ## Features
|
|
|
|
//!
|
|
|
|
//! * highly threaded code, uses Rust async
|
|
|
|
//! * static API definitions using schemas
|
|
|
|
//! * restartable systemd daemons using `systemd_notify`
|
|
|
|
//! * support for long running worker tasks (threads or async tokio tasks)
|
2021-10-01 04:43:30 +00:00
|
|
|
//! * supports separate access and authentication log files
|
|
|
|
//! * extra control socket to trigger management operations
|
2021-09-30 11:49:29 +00:00
|
|
|
//! - logfile rotation
|
|
|
|
//! - worker task management
|
|
|
|
//! * generic interface to authenticate user
|
|
|
|
|
2021-10-01 05:29:11 +00:00
|
|
|
use std::future::Future;
|
|
|
|
use std::pin::Pin;
|
2022-04-06 14:55:39 +00:00
|
|
|
use std::sync::atomic::{AtomicBool, Ordering};
|
2021-09-21 05:58:41 +00:00
|
|
|
|
|
|
|
use anyhow::{bail, format_err, Error};
|
2021-10-05 09:01:05 +00:00
|
|
|
use http::request::Parts;
|
|
|
|
use http::HeaderMap;
|
2022-04-06 14:55:39 +00:00
|
|
|
use hyper::{Body, Method, Response};
|
|
|
|
use nix::unistd::Pid;
|
2021-09-21 05:58:41 +00:00
|
|
|
|
2022-04-06 14:55:39 +00:00
|
|
|
use proxmox_router::UserInformation;
|
2021-11-23 16:57:00 +00:00
|
|
|
use proxmox_sys::fd::Fd;
|
|
|
|
use proxmox_sys::fs::CreateOptions;
|
2022-04-06 14:55:39 +00:00
|
|
|
use proxmox_sys::linux::procfs::PidStat;
|
2021-09-21 05:58:41 +00:00
|
|
|
|
2021-09-21 05:58:44 +00:00
|
|
|
mod compression;
|
|
|
|
pub use compression::*;
|
|
|
|
|
2021-09-21 05:58:41 +00:00
|
|
|
pub mod daemon;
|
2021-09-27 10:59:06 +00:00
|
|
|
|
2021-09-21 05:58:43 +00:00
|
|
|
pub mod formatter;
|
2021-09-21 05:58:40 +00:00
|
|
|
|
2021-09-21 05:58:42 +00:00
|
|
|
mod environment;
|
|
|
|
pub use environment::*;
|
|
|
|
|
2021-09-21 05:58:40 +00:00
|
|
|
mod state;
|
|
|
|
pub use state::*;
|
|
|
|
|
|
|
|
mod command_socket;
|
|
|
|
pub use command_socket::*;
|
|
|
|
|
|
|
|
mod file_logger;
|
2022-04-06 14:55:39 +00:00
|
|
|
pub use file_logger::{FileLogOptions, FileLogger};
|
2021-09-21 05:58:40 +00:00
|
|
|
|
|
|
|
mod api_config;
|
|
|
|
pub use api_config::ApiConfig;
|
|
|
|
|
2021-09-21 05:58:51 +00:00
|
|
|
mod rest;
|
2021-09-23 10:38:09 +00:00
|
|
|
pub use rest::RestServer;
|
2021-09-21 05:58:51 +00:00
|
|
|
|
2021-09-23 08:09:19 +00:00
|
|
|
mod worker_task;
|
|
|
|
pub use worker_task::*;
|
|
|
|
|
2021-09-23 10:38:09 +00:00
|
|
|
mod h2service;
|
|
|
|
pub use h2service::*;
|
|
|
|
|
2021-10-01 04:43:30 +00:00
|
|
|
/// Authentication Error
|
2021-09-21 05:58:40 +00:00
|
|
|
pub enum AuthError {
|
|
|
|
Generic(Error),
|
|
|
|
NoData,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Error> for AuthError {
|
|
|
|
fn from(err: Error) -> Self {
|
|
|
|
AuthError::Generic(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-05 09:01:05 +00:00
|
|
|
/// User Authentication and index/root page generation methods
|
|
|
|
pub trait ServerAdapter: Send + Sync {
|
|
|
|
/// Returns the index/root page
|
|
|
|
fn get_index(
|
|
|
|
&self,
|
|
|
|
rest_env: RestEnvironment,
|
|
|
|
parts: Parts,
|
|
|
|
) -> Pin<Box<dyn Future<Output = Response<Body>> + Send>>;
|
|
|
|
|
2021-09-30 08:33:57 +00:00
|
|
|
/// Extract user credentials from headers and check them.
|
|
|
|
///
|
|
|
|
/// If credenthials are valid, returns the username and a
|
|
|
|
/// [UserInformation] object to query additional user data.
|
2021-10-01 05:29:11 +00:00
|
|
|
fn check_auth<'a>(
|
|
|
|
&'a self,
|
2021-10-05 09:01:05 +00:00
|
|
|
headers: &'a HeaderMap,
|
|
|
|
method: &'a Method,
|
2022-04-06 14:55:39 +00:00
|
|
|
) -> Pin<
|
|
|
|
Box<
|
|
|
|
dyn Future<Output = Result<(String, Box<dyn UserInformation + Sync + Send>), AuthError>>
|
|
|
|
+ Send
|
|
|
|
+ 'a,
|
|
|
|
>,
|
|
|
|
>;
|
2021-09-21 05:58:40 +00:00
|
|
|
}
|
|
|
|
|
2022-04-06 14:55:39 +00:00
|
|
|
lazy_static::lazy_static! {
|
2021-09-23 08:09:19 +00:00
|
|
|
static ref PID: i32 = unsafe { libc::getpid() };
|
|
|
|
static ref PSTART: u64 = PidStat::read_from_pid(Pid::from_raw(*PID)).unwrap().starttime;
|
|
|
|
}
|
|
|
|
|
2021-09-30 08:33:57 +00:00
|
|
|
/// Retruns the current process ID (see [libc::getpid])
|
|
|
|
///
|
|
|
|
/// The value is cached at startup (so it is invalid after a fork)
|
2021-09-30 09:59:21 +00:00
|
|
|
pub(crate) fn pid() -> i32 {
|
2021-09-23 08:09:19 +00:00
|
|
|
*PID
|
|
|
|
}
|
|
|
|
|
2021-09-30 08:33:57 +00:00
|
|
|
/// Returns the starttime of the process (see [PidStat])
|
|
|
|
///
|
|
|
|
/// The value is cached at startup (so it is invalid after a fork)
|
2021-09-30 09:59:21 +00:00
|
|
|
pub(crate) fn pstart() -> u64 {
|
2021-09-23 08:09:19 +00:00
|
|
|
*PSTART
|
|
|
|
}
|
|
|
|
|
2021-09-30 08:33:57 +00:00
|
|
|
/// Helper to write the PID into a file
|
2021-09-23 08:09:19 +00:00
|
|
|
pub fn write_pid(pid_fn: &str) -> Result<(), Error> {
|
|
|
|
let pid_str = format!("{}\n", *PID);
|
2021-11-23 16:57:00 +00:00
|
|
|
proxmox_sys::fs::replace_file(pid_fn, pid_str.as_bytes(), CreateOptions::new(), false)
|
2021-09-23 08:09:19 +00:00
|
|
|
}
|
|
|
|
|
2021-09-30 08:33:57 +00:00
|
|
|
/// Helper to read the PID from a file
|
2021-09-23 08:09:19 +00:00
|
|
|
pub fn read_pid(pid_fn: &str) -> Result<i32, Error> {
|
2021-11-23 16:57:00 +00:00
|
|
|
let pid = proxmox_sys::fs::file_get_contents(pid_fn)?;
|
2021-09-23 08:09:19 +00:00
|
|
|
let pid = std::str::from_utf8(&pid)?.trim();
|
2022-04-06 14:55:39 +00:00
|
|
|
pid.parse()
|
|
|
|
.map_err(|err| format_err!("could not parse pid - {}", err))
|
2021-09-23 08:09:19 +00:00
|
|
|
}
|
|
|
|
|
2021-09-30 08:33:57 +00:00
|
|
|
/// Returns the control socket path for a specific process ID.
|
|
|
|
///
|
|
|
|
/// Note: The control socket always uses @/run/proxmox-backup/ as
|
|
|
|
/// prefix for historic reason. This does not matter because the
|
|
|
|
/// generated path is unique for each ``pid`` anyways.
|
2021-09-23 08:09:19 +00:00
|
|
|
pub fn ctrl_sock_from_pid(pid: i32) -> String {
|
|
|
|
// Note: The control socket always uses @/run/proxmox-backup/ as prefix
|
|
|
|
// for historc reason.
|
|
|
|
format!("\0{}/control-{}.sock", "/run/proxmox-backup", pid)
|
|
|
|
}
|
|
|
|
|
2021-09-30 08:33:57 +00:00
|
|
|
/// Returns the control socket path for this server.
|
2021-09-23 08:09:19 +00:00
|
|
|
pub fn our_ctrl_sock() -> String {
|
|
|
|
ctrl_sock_from_pid(*PID)
|
|
|
|
}
|
|
|
|
|
2021-09-29 10:46:00 +00:00
|
|
|
static SHUTDOWN_REQUESTED: AtomicBool = AtomicBool::new(false);
|
2021-09-21 05:58:40 +00:00
|
|
|
|
2021-09-30 08:33:57 +00:00
|
|
|
/// Request a server shutdown (usually called from [catch_shutdown_signal])
|
2021-09-21 05:58:40 +00:00
|
|
|
pub fn request_shutdown() {
|
2021-09-29 10:46:00 +00:00
|
|
|
SHUTDOWN_REQUESTED.store(true, Ordering::SeqCst);
|
2021-09-21 05:58:40 +00:00
|
|
|
crate::server_shutdown();
|
|
|
|
}
|
|
|
|
|
2021-09-30 08:33:57 +00:00
|
|
|
/// Returns true if there was a shutdown request.
|
2021-09-21 05:58:40 +00:00
|
|
|
#[inline(always)]
|
|
|
|
pub fn shutdown_requested() -> bool {
|
2021-09-29 10:46:00 +00:00
|
|
|
SHUTDOWN_REQUESTED.load(Ordering::SeqCst)
|
2021-09-21 05:58:40 +00:00
|
|
|
}
|
|
|
|
|
2021-09-30 08:33:57 +00:00
|
|
|
/// Raise an error if there was a shutdown request.
|
2021-09-21 05:58:40 +00:00
|
|
|
pub fn fail_on_shutdown() -> Result<(), Error> {
|
|
|
|
if shutdown_requested() {
|
|
|
|
bail!("Server shutdown requested - aborting task");
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2021-09-21 05:58:41 +00:00
|
|
|
/// safe wrapper for `nix::sys::socket::socketpair` defaulting to `O_CLOEXEC` and guarding the file
|
|
|
|
/// descriptors.
|
|
|
|
pub fn socketpair() -> Result<(Fd, Fd), Error> {
|
|
|
|
use nix::sys::socket;
|
|
|
|
let (pa, pb) = socket::socketpair(
|
|
|
|
socket::AddressFamily::Unix,
|
|
|
|
socket::SockType::Stream,
|
|
|
|
None,
|
|
|
|
socket::SockFlag::SOCK_CLOEXEC,
|
|
|
|
)?;
|
|
|
|
Ok((Fd(pa), Fd(pb)))
|
|
|
|
}
|
|
|
|
|
2021-09-21 05:58:45 +00:00
|
|
|
/// Extract a specific cookie from cookie header.
|
|
|
|
/// We assume cookie_name is already url encoded.
|
|
|
|
pub fn extract_cookie(cookie: &str, cookie_name: &str) -> Option<String> {
|
|
|
|
for pair in cookie.split(';') {
|
|
|
|
let (name, value) = match pair.find('=') {
|
|
|
|
Some(i) => (pair[..i].trim(), pair[(i + 1)..].trim()),
|
|
|
|
None => return None, // Cookie format error
|
|
|
|
};
|
|
|
|
|
|
|
|
if name == cookie_name {
|
|
|
|
use percent_encoding::percent_decode;
|
|
|
|
if let Ok(value) = percent_decode(value.as_bytes()).decode_utf8() {
|
|
|
|
return Some(value.into());
|
|
|
|
} else {
|
|
|
|
return None; // Cookie format error
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
2022-01-31 14:50:23 +00:00
|
|
|
/// Extract a specific cookie from a HeaderMap's "COOKIE" entry.
|
|
|
|
/// We assume cookie_name is already url encoded.
|
|
|
|
pub fn cookie_from_header(headers: &http::HeaderMap, cookie_name: &str) -> Option<String> {
|
|
|
|
if let Some(Ok(cookie)) = headers.get("COOKIE").map(|v| v.to_str()) {
|
2022-02-08 13:57:16 +00:00
|
|
|
extract_cookie(cookie, cookie_name)
|
2022-01-31 14:50:23 +00:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-21 05:58:45 +00:00
|
|
|
/// normalize uri path
|
|
|
|
///
|
|
|
|
/// Do not allow ".", "..", or hidden files ".XXXX"
|
|
|
|
/// Also remove empty path components
|
|
|
|
pub fn normalize_uri_path(path: &str) -> Result<(String, Vec<&str>), Error> {
|
|
|
|
let items = path.split('/');
|
|
|
|
|
|
|
|
let mut path = String::new();
|
|
|
|
let mut components = vec![];
|
|
|
|
|
|
|
|
for name in items {
|
|
|
|
if name.is_empty() {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if name.starts_with('.') {
|
|
|
|
bail!("Path contains illegal components.");
|
|
|
|
}
|
|
|
|
path.push('/');
|
|
|
|
path.push_str(name);
|
|
|
|
components.push(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok((path, components))
|
|
|
|
}
|