shitbox/corebox/commands/sync/mod.rs

59 lines
1.9 KiB
Rust

use super::Cmd;
use clap::{Arg, ArgAction, Command};
use std::{error::Error, fs::OpenOptions};
#[derive(Debug)]
pub struct Sync;
impl Cmd for Sync {
fn cli(&self) -> clap::Command {
Command::new("sync")
.about("force completion of pending disk writes (flush cache)")
.author("Nathan Fisher")
.version(env!("CARGO_PKG_VERSION"))
.args([
Arg::new("data")
.short('d')
.long("data")
.help("sync only file data, no unneeded metadata")
.conflicts_with("fs")
.action(ArgAction::SetTrue),
Arg::new("fs")
.short('f')
.long("file-system")
.help("sync the file systems that contain the files")
.action(ArgAction::SetTrue),
Arg::new("FILE")
.help(
"If one or more files are specified, sync only them, or \
their containing file systems.",
)
.num_args(0..),
])
}
fn run(&self, matches: &clap::ArgMatches) -> Result<(), Box<dyn Error>> {
if let Some(files) = matches.get_many::<String>("FILE") {
for f in files {
let mut opts = OpenOptions::new();
let opts = opts.read(true).write(true);
let fd = opts.open(f)?;
if matches.get_flag("data") {
unistd::fdatasync(&fd)?;
} else if matches.get_flag("fs") {
unistd::syncfs(&fd)?;
} else {
unistd::fsync(&fd)?;
}
}
} else {
unistd::sync();
}
Ok(())
}
fn path(&self) -> Option<shitbox::Path> {
Some(shitbox::Path::Bin)
}
}