1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
use nix::{
fcntl::{open, OFlag},
sys::stat::Mode,
unistd::{close, fsync, unlink, write},
};
use std::{path::Path, fmt};
#[derive(Debug)]
pub struct Error {
message: String,
}
impl<T> From<T> for Error
where
T: std::error::Error,
{
fn from(e: T) -> Self {
Error {
message: format!("{}", e),
}
}
}
impl std::fmt::Display for Error {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "{}", self)
}
}
pub fn atomic_write(path: impl Into<String>, contents: impl Into<Vec<u8>>) -> Result<(), Error> {
let original_path = path.into();
let mut tmp_path = original_path.clone();
tmp_path.push_str(".tmp");
let tmp_path_s = Path::new(&tmp_path);
let original_path_s = Path::new(&original_path);
match unlink(tmp_path_s) {
Ok(()) | Err(nix::Error::Sys(nix::errno::Errno::ENOENT)) => {}
Err(e) => return Err(e.into()),
}
let fd = open(
tmp_path_s,
OFlag::O_RDWR | OFlag::O_CREAT | OFlag::O_TRUNC,
Mode::S_IRUSR
| Mode::S_IWUSR
| Mode::S_IRGRP
| Mode::S_IWGRP
| Mode::S_IROTH
| Mode::S_IWOTH,
)?;
write(fd, &contents.into())?;
fsync(fd)?;
close(fd)?;
std::fs::rename(tmp_path_s, original_path_s)?;
Ok(())
}