#async #disk #back-end #facade #fs #floppy

floppy-disk

async filesystem facade for rust!

17 releases

0.2.6 Jul 19, 2023
0.2.5 Jun 22, 2023
0.2.1 May 30, 2023
0.1.8 May 29, 2023
0.0.0 Mar 16, 2023

#365 in Filesystem

Download history 35/week @ 2023-12-09 151/week @ 2023-12-16 14/week @ 2023-12-23 140/week @ 2023-12-30 16/week @ 2024-01-06 5/week @ 2024-01-13 38/week @ 2024-01-27 15/week @ 2024-02-03 12/week @ 2024-02-10 74/week @ 2024-02-17 59/week @ 2024-02-24 74/week @ 2024-03-02 41/week @ 2024-03-09 56/week @ 2024-03-16 119/week @ 2024-03-23

295 downloads per month
Used in 5 crates

MIT license

51KB
1.5K SLoC

floppy-disk

floppy disk is a WIP, async-only filesystem facade for Rust.

What?

Have you ever worked with std::fs? tokio::fs? Then you've probably realised that testing filesystem code is difficult and sometimes scary. Is that fs::remove_dir_all really safe to run?

The point of floppy disk is to fix this. Rather than always using the real filesystem, floppy disk lets you choose a backend for your filesystem access, via the FloppyDisk trait. Current implementations include in-memory and real filesystem via Tokio. This way, you can use the real filesystem when you need, but have your tests hit a fake in-memory filesystem instead.

Features

  • Pluggable filesystem backends
    • In-memory (WIP)
    • Tokio
  • Write-your-own with the FloppyDisk trait
  • Fully-async
    • Light evil involved

Caveats

  • floppy disk is a 0.x.y project! You probably don't want to use it in production.
  • async-only! There is some small bridging to sync code, like MemFile implementing Read/Write/Seek, but this is mostly a hack to make working with sync-only external libraries (ex. ar) easier.
  • in-memory fs may not be performant-enough

Example usage

floppy disk attempts to recreate the std::fs API 1:1, with the caveat of being async-only.

let fs = ...; // MemFloppyDisk::new() | TokioFloppyDisk::new()
fs.create_dir_all("/foo/bar").await?;
fs.write("/foo/bar/baz.txt", b"hello world").await?;
let contents = fs.read_to_string("/foo/bar/baz.txt").await?;
assert_eq!(contents, "hello world");

Passing a FloppyDisk around:

struct MyStruct<'a, F: FloppyDisk<'a>> {
    fs: F,
    _marker: PhantomData<&'a ()>,
}

async fn my_fn<'a, F: FloppyDisk<'a>> {
   // ...
}

Dependencies

~6–14MB
~142K SLoC