Add ZipFile::is_dir() and ZipFile::is_file().

The naming matches that of std::fs::Metadata.

An entry is determined to be a directory based on the presence of
a trailing path separator, i.e. '/' or '\'.

This patch adds a small test zip containing files and directories.
Their names match their type so as to make testing easy.
This commit is contained in:
Damien Collard 2019-01-27 15:57:30 +01:00
parent 659c9186c6
commit b9af51e654
2 changed files with 28 additions and 0 deletions

View file

@ -465,6 +465,14 @@ impl<'a> ZipFile<'a> {
pub fn last_modified(&self) -> DateTime { pub fn last_modified(&self) -> DateTime {
self.data.last_modified_time self.data.last_modified_time
} }
/// Returns whether the file is actually a directory
pub fn is_dir(&self) -> bool {
self.name().chars().rev().next().map_or(false, |c| c == '/' || c == '\\')
}
/// Returns whether the file is a regular file
pub fn is_file(&self) -> bool {
!self.is_dir()
}
/// Get unix mode for the file /// Get unix mode for the file
pub fn unix_mode(&self) -> Option<u32> { pub fn unix_mode(&self) -> Option<u32> {
if self.data.external_attributes == 0 { if self.data.external_attributes == 0 {
@ -711,4 +719,24 @@ mod test {
assert_eq!(buf3, buf4); assert_eq!(buf3, buf4);
assert!(buf1 != buf3); assert!(buf1 != buf3);
} }
#[test]
fn file_and_dir_predicates() {
use super::ZipArchive;
use std::io;
let mut v = Vec::new();
v.extend_from_slice(include_bytes!("../tests/data/files_and_dirs.zip"));
let mut zip = ZipArchive::new(io::Cursor::new(v)).unwrap();
for i in 0..zip.len() {
let zip_file = zip.by_index(i).unwrap();
let full_name = zip_file.sanitized_name();
let file_name = full_name.file_name().unwrap().to_str().unwrap();
assert!(
(file_name.starts_with("dir") && zip_file.is_dir())
|| (file_name.starts_with("file") && zip_file.is_file())
);
}
}
} }

Binary file not shown.