2023-01-14 10:36:08 -05:00
|
|
|
use super::Cmd;
|
2023-01-21 18:25:09 -05:00
|
|
|
use crate::args;
|
|
|
|
use clap::{Arg, Command};
|
2023-01-15 09:22:02 -05:00
|
|
|
use std::{ffi::CString, io, process};
|
2023-01-14 10:36:08 -05:00
|
|
|
|
|
|
|
#[derive(Debug, Default)]
|
|
|
|
pub struct Unlink;
|
|
|
|
|
|
|
|
impl Cmd for Unlink {
|
|
|
|
fn cli(&self) -> clap::Command {
|
|
|
|
Command::new("unlink")
|
|
|
|
.about("call the unlink function to remove the specified file")
|
|
|
|
.author("Nathan Fisher")
|
|
|
|
.version(env!("CARGO_PKG_VERSION"))
|
|
|
|
.args([
|
2023-01-21 18:25:09 -05:00
|
|
|
args::verbose(),
|
2023-01-14 10:36:08 -05:00
|
|
|
Arg::new("file")
|
|
|
|
.value_name("FILE")
|
|
|
|
.required(true)
|
|
|
|
.num_args(1..),
|
|
|
|
])
|
|
|
|
}
|
|
|
|
|
|
|
|
fn run(&self, matches: Option<&clap::ArgMatches>) -> Result<(), Box<dyn std::error::Error>> {
|
|
|
|
let Some(matches) = matches else {
|
|
|
|
return Err(Box::new(io::Error::new(io::ErrorKind::Other, "no input")));
|
|
|
|
};
|
|
|
|
if let Some(files) = matches.get_many::<String>("file") {
|
|
|
|
for f in files {
|
|
|
|
let fname = CString::new(f.as_str())?;
|
|
|
|
let ret = unsafe { libc::unlink(fname.as_ptr()) };
|
|
|
|
if matches.get_flag("verbose") && ret == 0 {
|
|
|
|
println!("unlink: {f}");
|
|
|
|
} else if ret != 0 {
|
|
|
|
let err = io::Error::last_os_error();
|
|
|
|
eprintln!("unlink: cannot unlink {f}: {err}");
|
|
|
|
process::exit(ret);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn path(&self) -> Option<crate::Path> {
|
|
|
|
Some(crate::Path::UsrBin)
|
|
|
|
}
|
|
|
|
}
|