feat: provide archive extraction API

This commit is contained in:
Marli Frost 2020-09-12 10:38:47 +01:00
parent a35c8ffa91
commit d0e905acc5
No known key found for this signature in database
GPG key ID: CB0BEA7CF9BD1245

View file

@ -311,6 +311,49 @@ impl<R: Read + io::Seek> ZipArchive<R> {
comment: footer.zip_file_comment, comment: footer.zip_file_comment,
}) })
} }
/// Extract a Zip archive into a directory.
///
/// Malformed and malicious paths are rejected so that they cannot escape
/// the given directory.
///
/// This bails on the first error and does not attempt cleanup.
///
/// # Platform-specific behaviour
///
/// On unix systems permissions from the zip file are preserved, if they exist.
pub fn extract<P: AsRef<Path>>(&mut self, directory: P) -> ZipResult<()> {
use std::fs;
for i in 0..self.len() {
let mut file = self.by_index(i)?;
let filepath = file
.name_as_child()
.ok_or(ZipError::InvalidArchive("Invalid file path"))?;
let outpath = directory.as_ref().join(filepath);
if (file.name()).ends_with('/') {
fs::create_dir_all(&outpath)?;
} else {
if let Some(p) = outpath.parent() {
if !p.exists() {
fs::create_dir_all(&p)?;
}
}
let mut outfile = fs::File::create(&outpath)?;
io::copy(&mut file, &mut outfile)?;
}
// Get and Set permissions
#[cfg(unix)]
{
use std::os::unix::fs::PermissionsExt;
if let Some(mode) = file.unix_mode() {
fs::set_permissions(&outpath, fs::Permissions::from_mode(mode))?;
}
}
}
Ok(())
}
/// Number of files contained in this zip. /// Number of files contained in this zip.
pub fn len(&self) -> usize { pub fn len(&self) -> usize {
@ -967,8 +1010,7 @@ mod test {
for i in 0..zip.len() { for i in 0..zip.len() {
let zip_file = zip.by_index(i).unwrap(); let zip_file = zip.by_index(i).unwrap();
#[allow(deprecated)] let full_name = zip_file.name_as_child().unwrap();
let full_name = zip_file.sanitized_name();
let file_name = full_name.file_name().unwrap().to_str().unwrap(); let file_name = full_name.file_name().unwrap().to_str().unwrap();
assert!( assert!(
(file_name.starts_with("dir") && zip_file.is_dir()) (file_name.starts_with("dir") && zip_file.is_dir())