feat(supervisor): rotating log writer
This commit is contained in:
@@ -2,10 +2,12 @@
|
|||||||
|
|
||||||
pub mod backoff;
|
pub mod backoff;
|
||||||
pub mod child;
|
pub mod child;
|
||||||
|
pub mod logs;
|
||||||
pub mod policy;
|
pub mod policy;
|
||||||
pub mod retry_window;
|
pub mod retry_window;
|
||||||
|
|
||||||
pub use backoff::Backoff;
|
pub use backoff::Backoff;
|
||||||
pub use child::{ChildHandle, MockChild, MockChildController};
|
pub use child::{ChildHandle, MockChild, MockChildController};
|
||||||
|
pub use logs::RotatingLogWriter;
|
||||||
pub use policy::{RestartDecision, decide};
|
pub use policy::{RestartDecision, decide};
|
||||||
pub use retry_window::RetryWindow;
|
pub use retry_window::RetryWindow;
|
||||||
|
|||||||
@@ -0,0 +1,104 @@
|
|||||||
|
use std::fs::{File, OpenOptions};
|
||||||
|
use std::io::Write;
|
||||||
|
use std::path::{Path, PathBuf};
|
||||||
|
|
||||||
|
pub struct RotatingLogWriter {
|
||||||
|
base: PathBuf,
|
||||||
|
max_bytes: u64,
|
||||||
|
keep: usize,
|
||||||
|
file: File,
|
||||||
|
written: u64,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl RotatingLogWriter {
|
||||||
|
pub fn open(base: &Path, max_bytes: u64, keep: usize) -> std::io::Result<Self> {
|
||||||
|
if let Some(parent) = base.parent() {
|
||||||
|
std::fs::create_dir_all(parent)?;
|
||||||
|
}
|
||||||
|
let file = OpenOptions::new().create(true).append(true).open(base)?;
|
||||||
|
let written = file.metadata()?.len();
|
||||||
|
Ok(Self {
|
||||||
|
base: base.to_path_buf(),
|
||||||
|
max_bytes,
|
||||||
|
keep,
|
||||||
|
file,
|
||||||
|
written,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn write_line(&mut self, tag: &str, line: &str) -> std::io::Result<()> {
|
||||||
|
let bytes = format!("{tag} {line}\n");
|
||||||
|
self.file.write_all(bytes.as_bytes())?;
|
||||||
|
self.written += bytes.len() as u64;
|
||||||
|
if self.written >= self.max_bytes {
|
||||||
|
self.rotate()?;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn rotate(&mut self) -> std::io::Result<()> {
|
||||||
|
// Drop the current handle by replacing with /dev/null briefly.
|
||||||
|
self.file = OpenOptions::new().read(true).open("/dev/null")?;
|
||||||
|
for i in (1..self.keep).rev() {
|
||||||
|
let src = self.gen_path(i);
|
||||||
|
let dst = self.gen_path(i + 1);
|
||||||
|
if src.exists() {
|
||||||
|
let _ = std::fs::rename(&src, &dst);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if self.base.exists() {
|
||||||
|
let _ = std::fs::rename(&self.base, self.gen_path(1));
|
||||||
|
}
|
||||||
|
self.file = OpenOptions::new()
|
||||||
|
.create(true)
|
||||||
|
.append(true)
|
||||||
|
.open(&self.base)?;
|
||||||
|
self.written = 0;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn gen_path(&self, n: usize) -> PathBuf {
|
||||||
|
let mut s = self.base.as_os_str().to_os_string();
|
||||||
|
s.push(format!(".{n}"));
|
||||||
|
PathBuf::from(s)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
use tempfile::tempdir;
|
||||||
|
use std::io::Read;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn writes_lines_with_tags() {
|
||||||
|
let dir = tempdir().unwrap();
|
||||||
|
let base = dir.path().join("x.log");
|
||||||
|
let mut w = RotatingLogWriter::open(&base, 1024, 3).unwrap();
|
||||||
|
w.write_line("[out]", "hello").unwrap();
|
||||||
|
w.write_line("[err]", "boom").unwrap();
|
||||||
|
let mut s = String::new();
|
||||||
|
File::open(&base)
|
||||||
|
.unwrap()
|
||||||
|
.read_to_string(&mut s)
|
||||||
|
.unwrap();
|
||||||
|
assert_eq!(s, "[out] hello\n[err] boom\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn rotates_at_threshold() {
|
||||||
|
let dir = tempdir().unwrap();
|
||||||
|
let base = dir.path().join("x.log");
|
||||||
|
let mut w = RotatingLogWriter::open(&base, 20, 3).unwrap();
|
||||||
|
for _ in 0..5 {
|
||||||
|
w.write_line("[out]", "0123456789").unwrap();
|
||||||
|
}
|
||||||
|
assert!(base.exists());
|
||||||
|
let rotated = dir.path().join("x.log.1");
|
||||||
|
assert!(
|
||||||
|
rotated.exists(),
|
||||||
|
"expected rotated file at {}",
|
||||||
|
rotated.display()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user