Support for external file attributes
This commit is contained in:
parent
d95c925ac2
commit
a16962cd2c
4 changed files with 94 additions and 15 deletions
|
@ -3,6 +3,9 @@ extern crate zip;
|
||||||
use std::io;
|
use std::io;
|
||||||
use std::fs;
|
use std::fs;
|
||||||
|
|
||||||
|
#[cfg(unix)]
|
||||||
|
use std::os::unix::fs::PermissionsExt;
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
std::process::exit(real_main());
|
std::process::exit(real_main());
|
||||||
}
|
}
|
||||||
|
@ -30,28 +33,51 @@ fn real_main() -> i32
|
||||||
if comment.len() > 0 { println!(" File comment: {}", comment); }
|
if comment.len() > 0 { println!(" File comment: {}", comment); }
|
||||||
}
|
}
|
||||||
|
|
||||||
create_directory(outpath.parent().unwrap_or(std::path::Path::new("")));
|
create_directory(outpath.parent().unwrap_or(std::path::Path::new("")), None);
|
||||||
|
|
||||||
|
let perms = convert_permissions(file.unix_mode());
|
||||||
|
|
||||||
if (&*file.name()).ends_with("/") {
|
if (&*file.name()).ends_with("/") {
|
||||||
create_directory(&outpath);
|
create_directory(&outpath, perms);
|
||||||
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
write_file(&mut file, &outpath);
|
write_file(&mut file, &outpath, perms);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_file(reader: &mut zip::read::ZipFile, outpath: &std::path::Path)
|
#[cfg(unix)]
|
||||||
|
fn convert_permissions(mode: Option<u32>) -> Option<fs::Permissions>
|
||||||
{
|
{
|
||||||
let mut outfile = fs::File::create(&outpath).unwrap();
|
match mode {
|
||||||
io::copy(reader, &mut outfile).unwrap();
|
Some(mode) => Some(fs::Permissions::from_mode(mode)),
|
||||||
|
None => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#[cfg(not(unix))]
|
||||||
|
fn convert_permissions(_mode: Option<u32>) -> Option<fs::Permissions>
|
||||||
|
{
|
||||||
|
None
|
||||||
}
|
}
|
||||||
|
|
||||||
fn create_directory(outpath: &std::path::Path)
|
fn write_file(file: &mut zip::read::ZipFile, outpath: &std::path::Path, perms: Option<fs::Permissions>)
|
||||||
|
{
|
||||||
|
let mut outfile = fs::File::create(&outpath).unwrap();
|
||||||
|
io::copy(file, &mut outfile).unwrap();
|
||||||
|
if let Some(perms) = perms {
|
||||||
|
fs::set_permissions(outpath, perms).unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn create_directory(outpath: &std::path::Path, perms: Option<fs::Permissions>)
|
||||||
{
|
{
|
||||||
fs::create_dir_all(&outpath).unwrap();
|
fs::create_dir_all(&outpath).unwrap();
|
||||||
|
if let Some(perms) = perms {
|
||||||
|
fs::set_permissions(outpath, perms).unwrap();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn sanitize_filename(filename: &str) -> std::path::PathBuf
|
fn sanitize_filename(filename: &str) -> std::path::PathBuf
|
||||||
|
|
50
src/read.rs
50
src/read.rs
|
@ -10,13 +10,18 @@ use std::collections::HashMap;
|
||||||
use flate2;
|
use flate2;
|
||||||
use flate2::FlateReadExt;
|
use flate2::FlateReadExt;
|
||||||
use podio::{ReadPodExt, LittleEndian};
|
use podio::{ReadPodExt, LittleEndian};
|
||||||
use types::ZipFileData;
|
use types::{ZipFileData, SYSTEM_MSDOS, SYSTEM_UNIX};
|
||||||
use cp437::FromCp437;
|
use cp437::FromCp437;
|
||||||
use msdos_time::{TmMsDosExt, MsDosDateTime};
|
use msdos_time::{TmMsDosExt, MsDosDateTime};
|
||||||
|
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
use bzip2::read::BzDecoder;
|
use bzip2::read::BzDecoder;
|
||||||
|
|
||||||
|
mod ffi {
|
||||||
|
pub const S_IFDIR: u32 = 0o0040000;
|
||||||
|
pub const S_IFREG: u32 = 0o0100000;
|
||||||
|
}
|
||||||
|
|
||||||
/// Wrapper for reading the contents of a ZIP file.
|
/// Wrapper for reading the contents of a ZIP file.
|
||||||
///
|
///
|
||||||
/// ```
|
/// ```
|
||||||
|
@ -181,8 +186,8 @@ fn central_header_to_zip_file<R: Read+io::Seek>(reader: &mut R) -> ZipResult<Zip
|
||||||
return Err(ZipError::InvalidArchive("Invalid Central Directory header"))
|
return Err(ZipError::InvalidArchive("Invalid Central Directory header"))
|
||||||
}
|
}
|
||||||
|
|
||||||
try!(reader.read_u16::<LittleEndian>());
|
let version_made_by = try!(reader.read_u16::<LittleEndian>());
|
||||||
try!(reader.read_u16::<LittleEndian>());
|
let _version_to_extract = try!(reader.read_u16::<LittleEndian>());
|
||||||
let flags = try!(reader.read_u16::<LittleEndian>());
|
let flags = try!(reader.read_u16::<LittleEndian>());
|
||||||
let encrypted = flags & 1 == 1;
|
let encrypted = flags & 1 == 1;
|
||||||
let is_utf8 = flags & (1 << 11) != 0;
|
let is_utf8 = flags & (1 << 11) != 0;
|
||||||
|
@ -195,9 +200,9 @@ fn central_header_to_zip_file<R: Read+io::Seek>(reader: &mut R) -> ZipResult<Zip
|
||||||
let file_name_length = try!(reader.read_u16::<LittleEndian>()) as usize;
|
let file_name_length = try!(reader.read_u16::<LittleEndian>()) as usize;
|
||||||
let extra_field_length = try!(reader.read_u16::<LittleEndian>()) as usize;
|
let extra_field_length = try!(reader.read_u16::<LittleEndian>()) as usize;
|
||||||
let file_comment_length = try!(reader.read_u16::<LittleEndian>()) as usize;
|
let file_comment_length = try!(reader.read_u16::<LittleEndian>()) as usize;
|
||||||
try!(reader.read_u16::<LittleEndian>());
|
let _disk_number = try!(reader.read_u16::<LittleEndian>());
|
||||||
try!(reader.read_u16::<LittleEndian>());
|
let _internal_file_attributes = try!(reader.read_u16::<LittleEndian>());
|
||||||
try!(reader.read_u32::<LittleEndian>());
|
let external_file_attributes = try!(reader.read_u32::<LittleEndian>());
|
||||||
let offset = try!(reader.read_u32::<LittleEndian>()) as u64;
|
let offset = try!(reader.read_u32::<LittleEndian>()) as u64;
|
||||||
let file_name_raw = try!(ReadPodExt::read_exact(reader, file_name_length));
|
let file_name_raw = try!(ReadPodExt::read_exact(reader, file_name_length));
|
||||||
let extra_field = try!(ReadPodExt::read_exact(reader, extra_field_length));
|
let extra_field = try!(ReadPodExt::read_exact(reader, extra_field_length));
|
||||||
|
@ -234,6 +239,8 @@ fn central_header_to_zip_file<R: Read+io::Seek>(reader: &mut R) -> ZipResult<Zip
|
||||||
// Construct the result
|
// Construct the result
|
||||||
let mut result = ZipFileData
|
let mut result = ZipFileData
|
||||||
{
|
{
|
||||||
|
system: (version_made_by >> 8) as u8,
|
||||||
|
version: version_made_by as u8,
|
||||||
encrypted: encrypted,
|
encrypted: encrypted,
|
||||||
compression_method: CompressionMethod::from_u16(compression_method),
|
compression_method: CompressionMethod::from_u16(compression_method),
|
||||||
last_modified_time: try!(::time::Tm::from_msdos(MsDosDateTime::new(last_mod_time, last_mod_date))),
|
last_modified_time: try!(::time::Tm::from_msdos(MsDosDateTime::new(last_mod_time, last_mod_date))),
|
||||||
|
@ -244,6 +251,7 @@ fn central_header_to_zip_file<R: Read+io::Seek>(reader: &mut R) -> ZipResult<Zip
|
||||||
file_comment: file_comment,
|
file_comment: file_comment,
|
||||||
header_start: offset,
|
header_start: offset,
|
||||||
data_start: data_start,
|
data_start: data_start,
|
||||||
|
external_attributes: external_file_attributes,
|
||||||
};
|
};
|
||||||
|
|
||||||
try!(parse_extra_field(&mut result, &*extra_field));
|
try!(parse_extra_field(&mut result, &*extra_field));
|
||||||
|
@ -280,6 +288,14 @@ impl<'a> ZipFile<'a> {
|
||||||
ZipFileReader::Bzip2(ref mut r) => r as &mut Read,
|
ZipFileReader::Bzip2(ref mut r) => r as &mut Read,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
/// Get compatibility of the file attribute information
|
||||||
|
pub fn system(&self) -> u8 {
|
||||||
|
self.data.system
|
||||||
|
}
|
||||||
|
/// Get the version of the file
|
||||||
|
pub fn version(&self) -> u8 {
|
||||||
|
self.data.version
|
||||||
|
}
|
||||||
/// Get the name of the file
|
/// Get the name of the file
|
||||||
pub fn name(&self) -> &str {
|
pub fn name(&self) -> &str {
|
||||||
&*self.data.file_name
|
&*self.data.file_name
|
||||||
|
@ -304,6 +320,28 @@ impl<'a> ZipFile<'a> {
|
||||||
pub fn last_modified(&self) -> ::time::Tm {
|
pub fn last_modified(&self) -> ::time::Tm {
|
||||||
self.data.last_modified_time
|
self.data.last_modified_time
|
||||||
}
|
}
|
||||||
|
/// Get mode for the file
|
||||||
|
pub fn unix_mode(&self) -> Option<u32> {
|
||||||
|
match self.data.system {
|
||||||
|
s if s == SYSTEM_UNIX => {
|
||||||
|
Some(self.data.external_attributes >> 16)
|
||||||
|
},
|
||||||
|
s if s == SYSTEM_MSDOS => {
|
||||||
|
// Interpret MSDOS directory bit
|
||||||
|
let mut mode = if 0x10 == (self.data.external_attributes & 0x10) {
|
||||||
|
ffi::S_IFDIR | 0o0775
|
||||||
|
} else {
|
||||||
|
ffi::S_IFREG | 0o0664
|
||||||
|
};
|
||||||
|
if 0x01 == (self.data.external_attributes & 0x01) {
|
||||||
|
// Read-only bit; strip write permissions
|
||||||
|
mode &= 0o0555;
|
||||||
|
}
|
||||||
|
Some(mode)
|
||||||
|
},
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> Read for ZipFile<'a> {
|
impl<'a> Read for ZipFile<'a> {
|
||||||
|
|
11
src/types.rs
11
src/types.rs
|
@ -2,9 +2,18 @@
|
||||||
|
|
||||||
use time;
|
use time;
|
||||||
|
|
||||||
|
pub const SYSTEM_MSDOS: u8 = 0;
|
||||||
|
pub const SYSTEM_UNIX: u8 = 3;
|
||||||
|
|
||||||
|
pub const DEFAULT_VERSION: u8 = 20;
|
||||||
|
|
||||||
/// Structure representing a ZIP file.
|
/// Structure representing a ZIP file.
|
||||||
pub struct ZipFileData
|
pub struct ZipFileData
|
||||||
{
|
{
|
||||||
|
/// Compatibility of the file attribute information
|
||||||
|
pub system: u8,
|
||||||
|
/// Specification version
|
||||||
|
pub version: u8,
|
||||||
/// True if the file is encrypted.
|
/// True if the file is encrypted.
|
||||||
pub encrypted: bool,
|
pub encrypted: bool,
|
||||||
/// Compression method used to store the file
|
/// Compression method used to store the file
|
||||||
|
@ -25,4 +34,6 @@ pub struct ZipFileData
|
||||||
pub header_start: u64,
|
pub header_start: u64,
|
||||||
/// Specifies where the compressed data of the file starts
|
/// Specifies where the compressed data of the file starts
|
||||||
pub data_start: u64,
|
pub data_start: u64,
|
||||||
|
/// External file attributes
|
||||||
|
pub external_attributes: u32,
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
//! Structs for creating a new zip archive
|
//! Structs for creating a new zip archive
|
||||||
|
|
||||||
use compression::CompressionMethod;
|
use compression::CompressionMethod;
|
||||||
use types::ZipFileData;
|
use types::{ZipFileData, SYSTEM_MSDOS, DEFAULT_VERSION};
|
||||||
use spec;
|
use spec;
|
||||||
use crc32;
|
use crc32;
|
||||||
use result::{ZipResult, ZipError};
|
use result::{ZipResult, ZipError};
|
||||||
|
@ -134,6 +134,8 @@ impl<W: Write+io::Seek> ZipWriter<W>
|
||||||
|
|
||||||
let mut file = ZipFileData
|
let mut file = ZipFileData
|
||||||
{
|
{
|
||||||
|
system: SYSTEM_MSDOS,
|
||||||
|
version: DEFAULT_VERSION,
|
||||||
encrypted: false,
|
encrypted: false,
|
||||||
compression_method: compression,
|
compression_method: compression,
|
||||||
last_modified_time: time::now(),
|
last_modified_time: time::now(),
|
||||||
|
@ -144,6 +146,7 @@ impl<W: Write+io::Seek> ZipWriter<W>
|
||||||
file_comment: String::new(),
|
file_comment: String::new(),
|
||||||
header_start: header_start,
|
header_start: header_start,
|
||||||
data_start: 0,
|
data_start: 0,
|
||||||
|
external_attributes: 0,
|
||||||
};
|
};
|
||||||
try!(write_local_file_header(writer, &file));
|
try!(write_local_file_header(writer, &file));
|
||||||
|
|
||||||
|
@ -319,7 +322,8 @@ impl<W: Write+io::Seek> GenericZipWriter<W>
|
||||||
fn write_local_file_header<T: Write>(writer: &mut T, file: &ZipFileData) -> ZipResult<()>
|
fn write_local_file_header<T: Write>(writer: &mut T, file: &ZipFileData) -> ZipResult<()>
|
||||||
{
|
{
|
||||||
try!(writer.write_u32::<LittleEndian>(spec::LOCAL_FILE_HEADER_SIGNATURE));
|
try!(writer.write_u32::<LittleEndian>(spec::LOCAL_FILE_HEADER_SIGNATURE));
|
||||||
try!(writer.write_u16::<LittleEndian>(20));
|
let version_made_by = (file.system as u16) << 8 | (file.version as u16);
|
||||||
|
try!(writer.write_u16::<LittleEndian>(version_made_by));
|
||||||
let flag = if !file.file_name.is_ascii() { 1u16 << 11 } else { 0 };
|
let flag = if !file.file_name.is_ascii() { 1u16 << 11 } else { 0 };
|
||||||
try!(writer.write_u16::<LittleEndian>(flag));
|
try!(writer.write_u16::<LittleEndian>(flag));
|
||||||
try!(writer.write_u16::<LittleEndian>(file.compression_method.to_u16()));
|
try!(writer.write_u16::<LittleEndian>(file.compression_method.to_u16()));
|
||||||
|
|
Loading…
Add table
Reference in a new issue