diff --git a/crates/nu-command/src/filesystem/save.rs b/crates/nu-command/src/filesystem/save.rs index c4ae00f593..67fd1a2271 100644 --- a/crates/nu-command/src/filesystem/save.rs +++ b/crates/nu-command/src/filesystem/save.rs @@ -422,7 +422,22 @@ fn prepare_path( fn open_file(path: &Path, span: Span, append: bool) -> Result { let file = match (append, path.exists()) { (true, true) => std::fs::OpenOptions::new().append(true).open(path), - _ => std::fs::File::create(path), + _ => { + // This is a temporary solution until `std::fs::File::create` is fixed on Windows (rust-lang/rust#134893) + // A TOCTOU problem exists here, which may cause wrong error message to be shown + #[cfg(target_os = "windows")] + if path.is_dir() { + // It should be `io::ErrorKind::IsADirectory` but it's not available in stable yet (1.83) + Err(io::Error::new( + io::ErrorKind::Unsupported, + "Is a directory (os error 21)", + )) + } else { + std::fs::File::create(path) + } + #[cfg(not(target_os = "windows"))] + std::fs::File::create(path) + } }; file.map_err(|e| ShellError::GenericError { diff --git a/crates/nu-command/tests/commands/save.rs b/crates/nu-command/tests/commands/save.rs index 6b77160bee..8c2ea535b7 100644 --- a/crates/nu-command/tests/commands/save.rs +++ b/crates/nu-command/tests/commands/save.rs @@ -525,3 +525,12 @@ fn parent_redirection_doesnt_affect_save() { assert_eq!(actual.trim_end(), ""); }) } + +#[test] +fn force_save_to_dir() { + let actual = nu!(cwd: "crates/nu-command/tests/commands", r#" + "aaa" | save -f .. + "#); + + assert!(actual.err.contains("Is a directory")); +}