diff --git a/src/read.rs b/src/read.rs index dd58f603..4a68d79e 100644 --- a/src/read.rs +++ b/src/read.rs @@ -85,7 +85,7 @@ pub(crate) mod zip_archive { #[cfg(feature = "lzma")] use crate::read::lzma::LzmaDecoder; use crate::result::ZipError::{InvalidPassword, UnsupportedArchive}; -use crate::spec::path_to_string; +use crate::spec::{is_dir, path_to_string}; use crate::types::ffi::S_IFLNK; use crate::unstable::LittleEndianReadExt; pub use zip_archive::ZipArchive; @@ -689,26 +689,28 @@ impl ZipArchive { if file.is_symlink() && (cfg!(unix) || cfg!(windows)) { let mut target = Vec::with_capacity(file.size() as usize); file.read_exact(&mut target)?; - let Ok(target) = try_utf8_to_os_string(target) else { - return Err(ZipError::InvalidArchive("Invalid UTF-8 as symlink target")); - }; #[cfg(unix)] { + use std::os::unix::ffi::OsStringExt; + let target = OsString::from_vec(target); let target_path = directory.as_ref().join(target); std::os::unix::fs::symlink(target_path, outpath.as_path())?; } #[cfg(windows)] { + let Ok(target) = String::from_utf8(target) else { + return Err(ZipError::InvalidArchive( + "Invalid UTF-8 as symlink target", + )); + }; + let target_is_dir_from_archive = + self.shared.files.contains_key(&target) && is_dir(&target); let target_internal_path: PathBuf = target.into(); let target_path = directory.as_ref().join(target_internal_path.clone()); - let target_is_dir = if let Ok(meta) = std::fs::metadata(&target_path) { + let target_is_dir = if target_is_dir_from_archive { + true + } else if let Ok(meta) = std::fs::metadata(&target_path) { meta.is_dir() - } else if let Some(target_in_archive) = - self.index_for_path(&target_internal_path) - { - let (_, target_in_archive) = - self.shared.files.get_index(target_in_archive).unwrap(); - target_in_archive.is_dir() } else { false }; @@ -930,12 +932,6 @@ impl ZipArchive { } } -#[cfg(unix)] -fn try_utf8_to_os_string(utf8_bytes: Vec) -> Result { - use std::os::unix::ffi::OsStringExt; - Ok(OsString::from_vec(utf8_bytes)) -} - #[cfg(windows)] fn try_utf8_to_os_string(utf8_bytes: Vec) -> Result { Ok(OsString::from(String::from_utf8(utf8_bytes)?)) @@ -1260,10 +1256,7 @@ impl<'a> ZipFile<'a> { } /// Returns whether the file is actually a directory pub fn is_dir(&self) -> bool { - self.name() - .chars() - .next_back() - .map_or(false, |c| c == '/' || c == '\\') + is_dir(self.name()) } /// Returns whether the file is actually a symbolic link diff --git a/src/spec.rs b/src/spec.rs index 89481faf..893228e4 100644 --- a/src/spec.rs +++ b/src/spec.rs @@ -227,6 +227,13 @@ impl Zip64CentralDirectoryEnd { } } +pub(crate) fn is_dir(filename: &str) -> bool { + filename + .chars() + .next_back() + .map_or(false, |c| c == '/' || c == '\\') +} + /// Converts a path to the ZIP format (forward-slash-delimited and normalized). pub(crate) fn path_to_string>(path: T) -> Box { let mut maybe_original = None; diff --git a/src/types.rs b/src/types.rs index 4dc66386..f1f0141f 100644 --- a/src/types.rs +++ b/src/types.rs @@ -16,6 +16,7 @@ pub(crate) mod ffi { use crate::extra_fields::ExtraField; use crate::result::DateTimeRangeError; +use crate::spec::is_dir; use crate::types::ffi::S_IFDIR; use crate::CompressionMethod; #[cfg(feature = "time")] @@ -364,10 +365,7 @@ pub struct ZipFileData { impl ZipFileData { #[allow(dead_code)] pub fn is_dir(&self) -> bool { - self.file_name - .chars() - .next_back() - .map_or(false, |c| c == '/' || c == '\\') + is_dir(&self.file_name) } pub fn file_name_sanitized(&self) -> PathBuf {