2021-02-17 11:16:00 +00:00
|
|
|
// Copyright © 2021 Intel Corporation
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
//
|
|
|
|
|
|
|
|
use std::borrow::Cow;
|
|
|
|
use std::collections::HashMap;
|
|
|
|
use std::fs::File;
|
2023-07-29 21:53:36 +00:00
|
|
|
use std::io;
|
2021-02-17 11:16:00 +00:00
|
|
|
use std::os::unix::io::AsRawFd;
|
2023-08-13 15:13:59 +00:00
|
|
|
use std::sync::Arc;
|
2021-02-17 11:16:00 +00:00
|
|
|
use std::time::{Duration, Instant};
|
|
|
|
|
2024-09-29 09:48:45 +00:00
|
|
|
use once_cell::sync::OnceCell;
|
|
|
|
use serde::Serialize;
|
|
|
|
|
2023-09-14 09:19:12 +00:00
|
|
|
static MONITOR: OnceCell<MonitorHandle> = OnceCell::new();
|
2021-02-17 11:16:00 +00:00
|
|
|
|
2023-07-29 21:53:36 +00:00
|
|
|
#[derive(Serialize)]
|
|
|
|
struct Event<'a> {
|
|
|
|
timestamp: Duration,
|
|
|
|
source: &'a str,
|
|
|
|
event: &'a str,
|
|
|
|
properties: Option<&'a HashMap<Cow<'a, str>, Cow<'a, str>>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Monitor {
|
|
|
|
pub rx: flume::Receiver<String>,
|
2023-08-13 20:32:44 +00:00
|
|
|
pub file: Option<File>,
|
2023-08-13 15:13:59 +00:00
|
|
|
pub broadcast: Vec<flume::Sender<Arc<String>>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Monitor {
|
2023-08-13 20:32:44 +00:00
|
|
|
pub fn new(rx: flume::Receiver<String>, file: Option<File>) -> Self {
|
2023-08-13 15:13:59 +00:00
|
|
|
Self {
|
|
|
|
rx,
|
|
|
|
file,
|
|
|
|
broadcast: vec![],
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn subscribe(&mut self) -> flume::Receiver<Arc<String>> {
|
|
|
|
let (tx, rx) = flume::unbounded();
|
|
|
|
self.broadcast.push(tx);
|
|
|
|
rx
|
|
|
|
}
|
2023-07-29 21:53:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
struct MonitorHandle {
|
|
|
|
tx: flume::Sender<String>,
|
|
|
|
start: Instant,
|
|
|
|
}
|
|
|
|
|
|
|
|
fn set_file_nonblocking(file: &File) -> io::Result<()> {
|
2021-02-17 11:16:00 +00:00
|
|
|
let fd = file.as_raw_fd();
|
2023-07-29 21:53:36 +00:00
|
|
|
|
2022-11-16 21:22:45 +00:00
|
|
|
// SAFETY: FFI call to configure the fd
|
2021-02-17 11:16:00 +00:00
|
|
|
let ret = unsafe {
|
|
|
|
let mut flags = libc::fcntl(fd, libc::F_GETFL);
|
|
|
|
flags |= libc::O_NONBLOCK;
|
|
|
|
libc::fcntl(fd, libc::F_SETFL, flags)
|
|
|
|
};
|
2023-07-29 21:53:36 +00:00
|
|
|
|
2021-02-17 11:16:00 +00:00
|
|
|
if ret < 0 {
|
2023-07-29 21:53:36 +00:00
|
|
|
Err(io::Error::last_os_error())
|
|
|
|
} else {
|
|
|
|
Ok(())
|
2021-02-17 11:16:00 +00:00
|
|
|
}
|
2023-07-29 21:53:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// This function must only be called once from the main thread before any threads
|
|
|
|
/// are created to avoid race conditions.
|
2023-08-13 20:32:44 +00:00
|
|
|
pub fn set_monitor(file: Option<File>) -> io::Result<Monitor> {
|
2023-09-14 09:19:12 +00:00
|
|
|
// There is only one caller of this function, so MONITOR is written to only once
|
|
|
|
assert!(MONITOR.get().is_none());
|
2023-07-29 21:53:36 +00:00
|
|
|
|
2023-08-13 20:32:44 +00:00
|
|
|
if let Some(ref file) = file {
|
|
|
|
set_file_nonblocking(file)?;
|
|
|
|
}
|
|
|
|
|
2023-07-29 21:53:36 +00:00
|
|
|
let (tx, rx) = flume::unbounded();
|
2023-08-13 15:13:59 +00:00
|
|
|
let monitor = Monitor::new(rx, file);
|
2023-07-29 21:53:36 +00:00
|
|
|
|
2023-09-14 09:19:12 +00:00
|
|
|
MONITOR.get_or_init(|| MonitorHandle {
|
|
|
|
tx,
|
|
|
|
start: Instant::now(),
|
|
|
|
});
|
2021-02-17 11:16:00 +00:00
|
|
|
|
2023-07-29 21:53:36 +00:00
|
|
|
Ok(monitor)
|
2021-02-17 11:16:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn event_log(source: &str, event: &str, properties: Option<&HashMap<Cow<str>, Cow<str>>>) {
|
2023-09-14 09:19:12 +00:00
|
|
|
// `MONITOR` is always in a valid state (None or Some), because it is set
|
|
|
|
// only once before any threads are spawned, and it's not mutated
|
2023-07-29 21:53:36 +00:00
|
|
|
// afterwards. This function only creates immutable references to `MONITOR`.
|
|
|
|
// Because `MONITOR.tx` is `Sync`, it's safe to share `MONITOR` across
|
|
|
|
// threads, making this function thread-safe.
|
2023-09-14 09:19:12 +00:00
|
|
|
if let Some(monitor_handle) = MONITOR.get().as_ref() {
|
2023-07-29 21:53:36 +00:00
|
|
|
let event = Event {
|
|
|
|
timestamp: monitor_handle.start.elapsed(),
|
2021-02-17 11:16:00 +00:00
|
|
|
source,
|
|
|
|
event,
|
|
|
|
properties,
|
|
|
|
};
|
2022-05-16 18:52:25 +00:00
|
|
|
|
2023-07-29 21:53:36 +00:00
|
|
|
if let Ok(event) = serde_json::to_string_pretty(&event) {
|
|
|
|
monitor_handle.tx.send(event).ok();
|
|
|
|
}
|
2021-02-17 11:16:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Through the use of Cow<'a, str> it is possible to use String as well as
|
|
|
|
&str as the parameters:
|
|
|
|
e.g.
|
|
|
|
event!("cpu_manager", "create_vcpu", "id", cpu_id.to_string());
|
|
|
|
*/
|
|
|
|
#[macro_export]
|
|
|
|
macro_rules! event {
|
|
|
|
($source:expr, $event:expr) => {
|
|
|
|
$crate::event_log($source, $event, None)
|
|
|
|
};
|
|
|
|
($source:expr, $event:expr, $($key:expr, $value:expr),*) => {
|
|
|
|
{
|
|
|
|
let mut properties = ::std::collections::HashMap::new();
|
|
|
|
$(
|
|
|
|
properties.insert($key.into(), $value.into());
|
|
|
|
)+
|
|
|
|
$crate::event_log($source, $event, Some(&properties))
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|