8 releases

0.2.2 Nov 3, 2024
0.2.1 Sep 23, 2024
0.1.4 Apr 25, 2024
0.1.3 Mar 1, 2024
0.1.0 Aug 22, 2023

#79 in Filesystem

Download history 16077/week @ 2024-09-26 12862/week @ 2024-10-03 13684/week @ 2024-10-10 14609/week @ 2024-10-17 12629/week @ 2024-10-24 11794/week @ 2024-10-31 11900/week @ 2024-11-07 13082/week @ 2024-11-14 14148/week @ 2024-11-21 15161/week @ 2024-11-28 16867/week @ 2024-12-05 16145/week @ 2024-12-12 12547/week @ 2024-12-19 5881/week @ 2024-12-26 11746/week @ 2025-01-02 10945/week @ 2025-01-09

43,760 downloads per month
Used in 22 crates (4 directly)

BSD-3-Clause

75KB
1K SLoC

Atomic Write File

Crate Documentation Build Status Crash Tests Status License

This is a Rust crate that offers functionality to write and overwrite files atomically, that is: without leaving the file in an intermediate state. Either the new contents of the files are written to the filesystem, or the old contents (if any) are preserved.

This crate implements two main structs: AtomicWriteFile and OpenOptions, which mimic the standard std::fs::File and std::fs::OpenOptions as much as possible.

This crate supports all major platforms, including: Unix systems, Windows, and WASI.

This crate is throughly tested to ensure the integrity of the files it creates. In particular, it ships with tests that simulate kernel panics on Linux and check the file contents afterwards. You can see those tests in action on the dedicated GitHub workflow.

Motivation and Example

Consider the following snippet of code to write a configuration file in JSON format:

use std::io::Write;
use std::fs::File;

let mut file = File::options()
                    .write(true)
                    .create(true)
                    .open("config.json")?;

writeln!(file, "{{")?;
writeln!(file, "  \"key1\": \"value1\",")?;
writeln!(file, "  \"key2\": \"value2\"")?;
writeln!(file, "}}")?;

This code opens a file named config.json, truncates its contents (if the file already existed), and writes the JSON content line-by-line.

If the code is interrupted before all of the writeln! calls are completed (because of a panic, or a signal is received, or the process is killed, or a filesystem error occurs), then the file will be left in a broken state: it will not contain valid JSON data, and the original contents (if any) will be lost.

AtomicWriteFile solves this problem by placing the new contents into the destination file only after it has been completely written to the filesystem. The snippet above can be rewritten using AtomicWriteFile instead of File as follows:

use std::io::Write;
use atomic_write_file::AtomicWriteFile;

let mut file = AtomicWriteFile::options()
                               .open("config.json")?;

writeln!(file, "{{")?;
writeln!(file, "  \"key1\": \"value1\",")?;
writeln!(file, "  \"key2\": \"value2\"")?;
writeln!(file, "}}")?;

file.commit()?;

Note that this code is almost the same as the original, except that it now uses AtomicWriteFile instead of File and there's an additional call to commit().

If the code is interrupted early, before the call to commit(), the original file config.json will be left untouched. Only if the new contents are fully written to the filesystem, config.json will get them.

Documentation

Reference, examples, internal details, and limitations are all available on docs.rs/atomic-write-file.

Dependencies

~2MB
~40K SLoC