##// END OF EJS Templates
rust-vfs: rename `open` to `open_write` and `open_read` to `open`...
rust-vfs: rename `open` to `open_write` and `open_read` to `open` `open` being read *and* write is surprising because it differs from the Rust stdlib where `std::fs::File::open` is read-only by default. More importantly, writing is more dangerous than reading, so let's make it more explicit.

File last commit:

r53064:7be39c51 default
r53191:645d247d default
Show More
logging.rs
109 lines | 3.8 KiB | application/rls-services+xml | RustLexer
Simon Sapin
rust: Add a log file rotation utility...
r47341 use crate::errors::{HgError, HgResultExt, IoErrorContext, IoResultExt};
Raphaël Gomès
hg-core: add a complete VFS...
r53064 use crate::vfs::{Vfs, VfsImpl};
Simon Sapin
rust: Add a log file rotation utility...
r47341 use std::io::Write;
Raphaël Gomès
hg-core: add a complete VFS...
r53064 use std::path::Path;
Simon Sapin
rust: Add a log file rotation utility...
r47341
/// An utility to append to a log file with the given name, and optionally
/// rotate it after it reaches a certain maximum size.
///
/// Rotation works by renaming "example.log" to "example.log.1", after renaming
/// "example.log.1" to "example.log.2" etc up to the given maximum number of
/// files.
pub struct LogFile<'a> {
Raphaël Gomès
rust: add Vfs trait...
r52761 vfs: VfsImpl,
Simon Sapin
rust: Add a log file rotation utility...
r47341 name: &'a str,
max_size: Option<u64>,
max_files: u32,
}
impl<'a> LogFile<'a> {
Raphaël Gomès
rust: add Vfs trait...
r52761 pub fn new(vfs: VfsImpl, name: &'a str) -> Self {
Simon Sapin
rust: Add a log file rotation utility...
r47341 Self {
vfs,
name,
max_size: None,
max_files: 0,
}
}
/// Rotate before writing to a log file that was already larger than the
/// given size, in bytes. `None` disables rotation.
pub fn max_size(mut self, value: Option<u64>) -> Self {
self.max_size = value;
self
}
/// Keep this many rotated files `{name}.1` up to `{name}.{max}`, in
/// addition to the original `{name}` file.
pub fn max_files(mut self, value: u32) -> Self {
self.max_files = value;
self
}
/// Append the given `bytes` as-is to the log file, after rotating if
/// needed.
///
/// No trailing newline is added. Make sure to include one in `bytes` if
/// desired.
pub fn write(&self, bytes: &[u8]) -> Result<(), HgError> {
let path = self.vfs.join(self.name);
let context = || IoErrorContext::WritingFile(path.clone());
let open = || {
std::fs::OpenOptions::new()
.create(true)
.append(true)
.open(&path)
.with_context(context)
};
let mut file = open()?;
if let Some(max_size) = self.max_size {
if file.metadata().with_context(context)?.len() >= max_size {
// For example with `max_files == 5`, the first iteration of
// this loop has `i == 4` and renames `{name}.4` to `{name}.5`.
// The last iteration renames `{name}.1` to
// `{name}.2`
for i in (1..self.max_files).rev() {
self.vfs
.rename(
Raphaël Gomès
hg-core: add a complete VFS...
r53064 Path::new(&format!("{}.{}", self.name, i)),
Path::new(&format!("{}.{}", self.name, i + 1)),
false,
Simon Sapin
rust: Add a log file rotation utility...
r47341 )
.io_not_found_as_none()?;
}
// Then rename `{name}` to `{name}.1`. This is the
// previously-opened `file`.
self.vfs
Raphaël Gomès
hg-core: add a complete VFS...
r53064 .rename(
Path::new(&self.name),
Path::new(&format!("{}.1", self.name)),
false,
)
Simon Sapin
rust: Add a log file rotation utility...
r47341 .io_not_found_as_none()?;
// Finally, create a new `{name}` file and replace our `file`
// handle.
file = open()?;
}
}
file.write_all(bytes).with_context(context)?;
file.sync_all().with_context(context)
}
}
#[test]
fn test_rotation() {
let temp = tempfile::tempdir().unwrap();
Raphaël Gomès
hg-core: add a complete VFS...
r53064 let vfs = VfsImpl::new(temp.path().to_owned(), false);
Raphaël Gomès
rust: add Vfs trait...
r52761 let logger = LogFile::new(vfs.clone(), "log")
.max_size(Some(3))
.max_files(2);
Simon Sapin
rust: Add a log file rotation utility...
r47341 logger.write(b"one\n").unwrap();
logger.write(b"two\n").unwrap();
logger.write(b"3\n").unwrap();
logger.write(b"four\n").unwrap();
logger.write(b"five\n").unwrap();
assert_eq!(vfs.read("log").unwrap(), b"five\n");
assert_eq!(vfs.read("log.1").unwrap(), b"3\nfour\n");
assert_eq!(vfs.read("log.2").unwrap(), b"two\n");
assert!(vfs.read("log.3").io_not_found_as_none().unwrap().is_none());
}