Make santized_name part of the library

It is a common pitfall to use the name as-is during extraction. Adding
this function may prevent some of the issues.

Resolves #65
This commit is contained in:
Mathijs van de Nes 2018-04-20 13:35:20 +02:00
parent ce4104d682
commit 5e8c95e6ed
3 changed files with 49 additions and 19 deletions

View file

@ -20,7 +20,7 @@ fn real_main() -> i32 {
for i in 0..archive.len() {
let mut file = archive.by_index(i).unwrap();
let outpath = sanitize_filename(file.name());
let outpath = file.sanitized_name();
{
let comment = file.comment();
@ -55,21 +55,3 @@ fn real_main() -> i32 {
}
return 0;
}
fn sanitize_filename(filename: &str) -> std::path::PathBuf {
let no_null_filename = match filename.find('\0') {
Some(index) => &filename[0..index],
None => filename,
};
std::path::Path::new(no_null_filename)
.components()
.filter(|component| match *component {
std::path::Component::Normal(..) => true,
_ => false,
})
.fold(std::path::PathBuf::new(), |mut path, ref cur| {
path.push(cur.as_os_str());
path
})
}

View file

@ -415,6 +415,11 @@ impl<'a> ZipFile<'a> {
pub fn name_raw(&self) -> &[u8] {
&*self.data.file_name_raw
}
/// Get the name of the file in a sanitized form. It truncates the name to the first NULL byte,
/// removes a leading '/' and removes '..' parts.
pub fn sanitized_name(&self) -> ::std::path::PathBuf {
self.data.file_name_sanitized()
}
/// Get the comment of the file
pub fn comment(&self) -> &str {
&*self.data.file_comment

View file

@ -61,6 +61,26 @@ pub struct ZipFileData
pub external_attributes: u32,
}
impl ZipFileData {
pub fn file_name_sanitized(&self) -> ::std::path::PathBuf {
let no_null_filename = match self.file_name.find('\0') {
Some(index) => &self.file_name[0..index],
None => &self.file_name,
};
::std::path::Path::new(no_null_filename)
.components()
.filter(|component| match *component {
::std::path::Component::Normal(..) => true,
_ => false,
})
.fold(::std::path::PathBuf::new(), |mut path, ref cur| {
path.push(cur.as_os_str());
path
})
}
}
#[cfg(test)]
mod test {
#[test]
@ -71,4 +91,27 @@ mod test {
assert_eq!(System::from_u8(0), System::Dos);
assert_eq!(System::from_u8(3), System::Unix);
}
#[test]
fn sanitize() {
use super::*;
let file_name = "/path/../../../../etc/./passwd\0/etc/shadow".to_string();
let data = ZipFileData {
system: System::Dos,
version_made_by: 0,
encrypted: false,
compression_method: ::compression::CompressionMethod::Stored,
last_modified_time: time::empty_tm(),
crc32: 0,
compressed_size: 0,
uncompressed_size: 0,
file_name: file_name.clone(),
file_name_raw: file_name.into_bytes(),
file_comment: String::new(),
header_start: 0,
data_start: 0,
external_attributes: 0,
};
assert_eq!(data.file_name_sanitized().to_str().unwrap(), "path/etc/passwd");
}
}