feat: Add is_symlink method

This commit is contained in:
Chris Hennick 2024-05-13 19:52:14 -07:00
parent 5f44a2c3ed
commit 3bf0301e39
No known key found for this signature in database
GPG key ID: DA47AABA4961C509
4 changed files with 19 additions and 3 deletions

View file

@ -87,6 +87,7 @@ use crate::result::ZipError::{InvalidPassword, UnsupportedArchive};
use crate::spec::path_to_string;
use crate::unstable::LittleEndianReadExt;
pub use zip_archive::ZipArchive;
use crate::types::ffi::S_IFLNK;
#[allow(clippy::large_enum_variant)]
pub(crate) enum CryptoReader<'a> {
@ -1210,9 +1211,14 @@ impl<'a> ZipFile<'a> {
.map_or(false, |c| c == '/' || c == '\\')
}
/// Returns whether the file is a regular file
/// Returns whether the file is actually a symbolic link
pub fn is_symlink(&self) -> bool {
self.unix_mode().is_some_and(|mode| mode & S_IFLNK == S_IFLNK)
}
/// Returns whether the file is a normal file (i.e. not a directory or symlink)
pub fn is_file(&self) -> bool {
!self.is_dir()
!self.is_dir() && !self.is_symlink()
}
/// Get unix mode for the file
@ -1594,4 +1600,12 @@ mod test {
let mut file = reader.by_index(0).unwrap();
assert_eq!(file.read(&mut decompressed).unwrap(), 12);
}
#[test]
fn test_is_symlink() {
let mut v = Vec::new();
v.extend_from_slice(include_bytes!("../tests/data/symlink.zip"));
let mut reader = ZipArchive::new(Cursor::new(v)).unwrap();
assert!(reader.by_index(0).unwrap().is_symlink())
}
}

View file

@ -11,6 +11,7 @@ use {crate::read::ZipFile, crate::write::FileOptions};
pub(crate) mod ffi {
pub const S_IFDIR: u32 = 0o0040000;
pub const S_IFREG: u32 = 0o0100000;
pub const S_IFLNK: u32 = 0o0120000;
}
use crate::extra_fields::ExtraField;

View file

@ -1,5 +1,6 @@
//! Types for creating ZIP archives
use crate::write::ffi::S_IFLNK;
#[cfg(feature = "aes-crypto")]
use crate::aes::AesWriter;
use crate::compression::CompressionMethod;
@ -1341,7 +1342,7 @@ impl<W: Write + Seek> ZipWriter<W> {
if options.permissions.is_none() {
options.permissions = Some(0o777);
}
*options.permissions.as_mut().unwrap() |= 0o120000;
*options.permissions.as_mut().unwrap() |= S_IFLNK;
// The symlink target is stored as file content. And compressing the target path
// likely wastes space. So always store.
options.compression_method = Stored;

BIN
tests/data/symlink.zip Normal file

Binary file not shown.