use std::{fs, future::Future};
use async_stream::stream;
use futures_core::Stream;
use tokio_uring::fs::{create_dir_all, remove_file};
use crate::{
disk::tokio_uring::TokioUringFile,
fs::{FileMeta, FileSystemTag, Fs, OpenOptions},
path::{path_to_local, Path},
Error, MaybeSend,
};
pub struct TokioUringFs;
impl Fs for TokioUringFs {
type File = TokioUringFile;
fn file_system(&self) -> FileSystemTag {
FileSystemTag::Local
}
async fn open_options(&self, path: &Path, options: OpenOptions) -> Result<Self::File, Error> {
let local_path = path_to_local(path)?;
let file = tokio_uring::fs::OpenOptions::new()
.read(options.read)
.write(options.write)
.create(options.create)
.truncate(options.truncate)
.open(&local_path)
.await?;
let stat = file.statx().await?;
Ok(TokioUringFile {
file: Some(file),
pos: stat.stx_size,
})
}
async fn create_dir_all(path: &Path) -> Result<(), Error> {
let path = path_to_local(path)?;
create_dir_all(path).await?;
Ok(())
}
async fn list(
&self,
path: &Path,
) -> Result<impl Stream<Item = Result<FileMeta, Error>>, Error> {
let path = path_to_local(path)?;
let dir = path.read_dir()?;
Ok(stream! {
for entry in dir {
let entry = entry?;
yield Ok(FileMeta { path: Path::from_filesystem_path(entry.path())?, size: entry.metadata()?.len() });
}
})
}
async fn remove(&self, path: &Path) -> Result<(), Error> {
let path = path_to_local(path)?;
Ok(remove_file(path).await?)
}
async fn copy(&self, from: &Path, to: &Path) -> Result<(), Error> {
let from = path_to_local(from)?;
let to = path_to_local(to)?;
fs::copy(&from, &to)?;
Ok(())
}
async fn link(&self, from: &Path, to: &Path) -> Result<(), Error> {
let from = path_to_local(from)?;
let to = path_to_local(to)?;
fs::hard_link(&from, &to)?;
Ok(())
}
}