1
Fork 0
mirror of https://github.com/RGBCube/uutils-coreutils synced 2025-07-28 03:27:44 +00:00

Merge pull request #7492 from bitspill/rm

rm: skip prompt when stdin is not interactive; Fix #7326
This commit is contained in:
bitspill 2025-04-27 16:54:16 -05:00 committed by GitHub
parent 07501be4ae
commit d957e64999
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
2 changed files with 105 additions and 19 deletions

View file

@ -8,6 +8,7 @@
use clap::{Arg, ArgAction, Command, builder::ValueParser, parser::ValueSource}; use clap::{Arg, ArgAction, Command, builder::ValueParser, parser::ValueSource};
use std::ffi::{OsStr, OsString}; use std::ffi::{OsStr, OsString};
use std::fs::{self, Metadata}; use std::fs::{self, Metadata};
use std::io::{IsTerminal, stdin};
use std::ops::BitOr; use std::ops::BitOr;
#[cfg(not(windows))] #[cfg(not(windows))]
use std::os::unix::ffi::OsStrExt; use std::os::unix::ffi::OsStrExt;
@ -68,6 +69,25 @@ pub struct Options {
pub dir: bool, pub dir: bool,
/// `-v`, `--verbose` /// `-v`, `--verbose`
pub verbose: bool, pub verbose: bool,
#[doc(hidden)]
/// `---presume-input-tty`
/// Always use `None`; GNU flag for testing use only
pub __presume_input_tty: Option<bool>,
}
impl Default for Options {
fn default() -> Self {
Self {
force: false,
interactive: InteractiveMode::PromptProtected,
one_fs: false,
preserve_root: true,
recursive: false,
dir: false,
verbose: false,
__presume_input_tty: None,
}
}
} }
const ABOUT: &str = help_about!("rm.md"); const ABOUT: &str = help_about!("rm.md");
@ -145,6 +165,11 @@ pub fn uumain(args: impl uucore::Args) -> UResult<()> {
recursive: matches.get_flag(OPT_RECURSIVE), recursive: matches.get_flag(OPT_RECURSIVE),
dir: matches.get_flag(OPT_DIR), dir: matches.get_flag(OPT_DIR),
verbose: matches.get_flag(OPT_VERBOSE), verbose: matches.get_flag(OPT_VERBOSE),
__presume_input_tty: if matches.get_flag(PRESUME_INPUT_TTY) {
Some(true)
} else {
None
},
}; };
if options.interactive == InteractiveMode::Once && (options.recursive || files.len() > 3) { if options.interactive == InteractiveMode::Once && (options.recursive || files.len() > 3) {
let msg: String = format!( let msg: String = format!(
@ -608,13 +633,15 @@ fn prompt_file(path: &Path, options: &Options) -> bool {
prompt_yes!("remove file {}?", path.quote()) prompt_yes!("remove file {}?", path.quote())
}; };
} }
prompt_file_permission_readonly(path) prompt_file_permission_readonly(path, options)
} }
fn prompt_file_permission_readonly(path: &Path) -> bool { fn prompt_file_permission_readonly(path: &Path, options: &Options) -> bool {
match fs::metadata(path) { let stdin_ok = options.__presume_input_tty.unwrap_or(false) || stdin().is_terminal();
Ok(_) if is_writable(path) => true, match (stdin_ok, fs::metadata(path), options.interactive) {
Ok(metadata) if metadata.len() == 0 => prompt_yes!( (false, _, InteractiveMode::PromptProtected) => true,
(_, Ok(_), _) if is_writable(path) => true,
(_, Ok(metadata), _) if metadata.len() == 0 => prompt_yes!(
"remove write-protected regular empty file {}?", "remove write-protected regular empty file {}?",
path.quote() path.quote()
), ),
@ -622,26 +649,29 @@ fn prompt_file_permission_readonly(path: &Path) -> bool {
} }
} }
// For directories finding if they are writable or not is a hassle. In Unix we can use the built-in rust crate to to check mode bits. But other os don't have something similar afaik // For directories finding if they are writable or not is a hassle. In Unix we can use the built-in rust crate to check mode bits. But other os don't have something similar afaik
// Most cases are covered by keep eye out for edge cases // Most cases are covered by keep eye out for edge cases
#[cfg(unix)] #[cfg(unix)]
fn handle_writable_directory(path: &Path, options: &Options, metadata: &Metadata) -> bool { fn handle_writable_directory(path: &Path, options: &Options, metadata: &Metadata) -> bool {
let stdin_ok = options.__presume_input_tty.unwrap_or(false) || stdin().is_terminal();
match ( match (
stdin_ok,
is_readable_metadata(metadata), is_readable_metadata(metadata),
is_writable_metadata(metadata), is_writable_metadata(metadata),
options.interactive, options.interactive,
) { ) {
(false, false, _) => prompt_yes!( (false, _, _, InteractiveMode::PromptProtected) => true,
(_, false, false, _) => prompt_yes!(
"attempt removal of inaccessible directory {}?", "attempt removal of inaccessible directory {}?",
path.quote() path.quote()
), ),
(false, true, InteractiveMode::Always) => prompt_yes!( (_, false, true, InteractiveMode::Always) => prompt_yes!(
"attempt removal of inaccessible directory {}?", "attempt removal of inaccessible directory {}?",
path.quote() path.quote()
), ),
(true, false, _) => prompt_yes!("remove write-protected directory {}?", path.quote()), (_, true, false, _) => prompt_yes!("remove write-protected directory {}?", path.quote()),
(_, _, InteractiveMode::Always) => prompt_yes!("remove directory {}?", path.quote()), (_, _, _, InteractiveMode::Always) => prompt_yes!("remove directory {}?", path.quote()),
(_, _, _) => true, (_, _, _, _) => true,
} }
} }
@ -666,12 +696,12 @@ fn handle_writable_directory(path: &Path, options: &Options, metadata: &Metadata
use std::os::windows::prelude::MetadataExt; use std::os::windows::prelude::MetadataExt;
use windows_sys::Win32::Storage::FileSystem::FILE_ATTRIBUTE_READONLY; use windows_sys::Win32::Storage::FileSystem::FILE_ATTRIBUTE_READONLY;
let not_user_writable = (metadata.file_attributes() & FILE_ATTRIBUTE_READONLY) != 0; let not_user_writable = (metadata.file_attributes() & FILE_ATTRIBUTE_READONLY) != 0;
if not_user_writable { let stdin_ok = options.__presume_input_tty.unwrap_or(false) || stdin().is_terminal();
prompt_yes!("remove write-protected directory {}?", path.quote()) match (stdin_ok, not_user_writable, options.interactive) {
} else if options.interactive == InteractiveMode::Always { (false, _, InteractiveMode::PromptProtected) => true,
prompt_yes!("remove directory {}?", path.quote()) (_, true, _) => prompt_yes!("remove write-protected directory {}?", path.quote()),
} else { (_, _, InteractiveMode::Always) => prompt_yes!("remove directory {}?", path.quote()),
true (_, _, _) => true,
} }
} }

View file

@ -579,6 +579,50 @@ fn test_rm_prompts() {
assert!(!at.dir_exists("a")); assert!(!at.dir_exists("a"));
} }
#[cfg(feature = "chmod")]
#[test]
fn test_rm_prompts_no_tty() {
// This test ensures InteractiveMode.PromptProtected proceeds silently with non-interactive stdin
use std::io::Write;
let scene = TestScenario::new(util_name!());
let at = &scene.fixtures;
at.mkdir("a/");
let file_1 = "a/empty";
let file_2 = "a/empty-no-write";
let file_3 = "a/f-no-write";
at.touch(file_1);
at.touch(file_2);
at.make_file(file_3)
.write_all(b"not-empty")
.expect("Couldn't write to a/f-no-write");
at.symlink_dir("a/empty-f", "a/slink");
at.symlink_dir(".", "a/slink-dot");
let dir_1 = "a/b/";
let dir_2 = "a/b-no-write/";
at.mkdir(dir_1);
at.mkdir(dir_2);
scene
.ccmd("chmod")
.arg("u-w")
.arg(file_3)
.arg(dir_2)
.arg(file_2)
.succeeds();
scene.ucmd().arg("-r").arg("a").succeeds().no_output();
assert!(!at.dir_exists("a"));
}
#[test] #[test]
fn test_rm_force_prompts_order() { fn test_rm_force_prompts_order() {
// Needed for talking with stdin on platforms where CRLF or LF matters // Needed for talking with stdin on platforms where CRLF or LF matters
@ -646,7 +690,13 @@ fn test_prompt_write_protected_yes() {
scene.ccmd("chmod").arg("0").arg(file_1).succeeds(); scene.ccmd("chmod").arg("0").arg(file_1).succeeds();
scene.ucmd().arg(file_1).pipe_in("y").succeeds(); scene
.ucmd()
.arg("---presume-input-tty")
.arg(file_1)
.pipe_in("y")
.succeeds()
.stderr_contains("rm: remove write-protected regular empty file");
assert!(!at.file_exists(file_1)); assert!(!at.file_exists(file_1));
} }
@ -661,7 +711,13 @@ fn test_prompt_write_protected_no() {
scene.ccmd("chmod").arg("0").arg(file_2).succeeds(); scene.ccmd("chmod").arg("0").arg(file_2).succeeds();
scene.ucmd().arg(file_2).pipe_in("n").succeeds(); scene
.ucmd()
.arg("---presume-input-tty")
.arg(file_2)
.pipe_in("n")
.succeeds()
.stderr_contains("rm: remove write-protected regular empty file");
assert!(at.file_exists(file_2)); assert!(at.file_exists(file_2));
} }