Added documentation and moved types to top
This commit is contained in:
parent
4d07f695de
commit
02059ce351
7 changed files with 106 additions and 9 deletions
|
@ -6,7 +6,7 @@ fn main()
|
||||||
let fname = Path::new(args[1].as_slice());
|
let fname = Path::new(args[1].as_slice());
|
||||||
let file = std::io::File::open(&fname);
|
let file = std::io::File::open(&fname);
|
||||||
|
|
||||||
let zipcontainer = zip::reader::ZipReader::new(file).unwrap();
|
let zipcontainer = zip::ZipReader::new(file).unwrap();
|
||||||
|
|
||||||
for file in zipcontainer.files()
|
for file in zipcontainer.files()
|
||||||
{
|
{
|
||||||
|
|
|
@ -11,11 +11,7 @@ fn doit() -> std::io::IoResult<()>
|
||||||
let fname = Path::new(args[1].as_slice());
|
let fname = Path::new(args[1].as_slice());
|
||||||
let file = std::io::File::create(&fname).unwrap();
|
let file = std::io::File::create(&fname).unwrap();
|
||||||
|
|
||||||
let mut zip = zip::writer::ZipWriter::new(file);
|
let mut zip = zip::ZipWriter::new(file);
|
||||||
|
|
||||||
|
|
||||||
// try!(zip.start_file(b"test", zip::types::Stored));
|
|
||||||
// try!(zip.write(b""));
|
|
||||||
|
|
||||||
try!(zip.start_file(b"test/readme.txt", zip::types::Stored));
|
try!(zip.start_file(b"test/readme.txt", zip::types::Stored));
|
||||||
try!(zip.write(b"Hello, World!\n"));
|
try!(zip.write(b"Hello, World!\n"));
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
//! Helper module to compute a CRC32 checksum
|
||||||
|
|
||||||
use std::io;
|
use std::io;
|
||||||
|
|
||||||
static CRC32_TABLE : [u32, ..256] = [
|
static CRC32_TABLE : [u32, ..256] = [
|
||||||
|
@ -46,6 +48,7 @@ static CRC32_TABLE : [u32, ..256] = [
|
||||||
0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, 0x2d02ef8d
|
0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, 0x2d02ef8d
|
||||||
];
|
];
|
||||||
|
|
||||||
|
/// Update the checksum original based upon the contents of buf.
|
||||||
pub fn crc32(original: u32, buf: &[u8]) -> u32
|
pub fn crc32(original: u32, buf: &[u8]) -> u32
|
||||||
{
|
{
|
||||||
let mut crc = original ^ !0u32;
|
let mut crc = original ^ !0u32;
|
||||||
|
@ -58,6 +61,7 @@ pub fn crc32(original: u32, buf: &[u8]) -> u32
|
||||||
return crc ^ !0u32;
|
return crc ^ !0u32;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Reader that validates the CRC32 when it reaches the EOF.
|
||||||
pub struct Crc32Reader<R>
|
pub struct Crc32Reader<R>
|
||||||
{
|
{
|
||||||
inner: R,
|
inner: R,
|
||||||
|
@ -67,6 +71,7 @@ pub struct Crc32Reader<R>
|
||||||
|
|
||||||
impl<R: Reader> Crc32Reader<R>
|
impl<R: Reader> Crc32Reader<R>
|
||||||
{
|
{
|
||||||
|
/// Get a new Crc32Reader which check the inner reader against checksum.
|
||||||
pub fn new(inner: R, checksum: u32) -> Crc32Reader<R>
|
pub fn new(inner: R, checksum: u32) -> Crc32Reader<R>
|
||||||
{
|
{
|
||||||
Crc32Reader
|
Crc32Reader
|
||||||
|
|
10
src/lib.rs
10
src/lib.rs
|
@ -1,13 +1,19 @@
|
||||||
|
//! A basic ZipReader/Writer crate
|
||||||
|
|
||||||
#![feature(phase)]
|
#![feature(phase)]
|
||||||
#![feature(unsafe_destructor)]
|
#![feature(unsafe_destructor)]
|
||||||
|
#![warn(missing_doc)]
|
||||||
|
|
||||||
#[phase(plugin, link)] extern crate log;
|
#[phase(plugin, link)] extern crate log;
|
||||||
extern crate time;
|
extern crate time;
|
||||||
extern crate flate2;
|
extern crate flate2;
|
||||||
|
|
||||||
|
pub use reader::ZipReader;
|
||||||
|
pub use writer::ZipWriter;
|
||||||
|
|
||||||
mod util;
|
mod util;
|
||||||
mod spec;
|
mod spec;
|
||||||
pub mod crc32;
|
pub mod crc32;
|
||||||
pub mod reader;
|
mod reader;
|
||||||
pub mod types;
|
pub mod types;
|
||||||
pub mod writer;
|
mod writer;
|
||||||
|
|
|
@ -7,6 +7,30 @@ use std::io::{IoResult, IoError};
|
||||||
use std::cell::RefCell;
|
use std::cell::RefCell;
|
||||||
use flate2::FlateReader;
|
use flate2::FlateReader;
|
||||||
|
|
||||||
|
/// Wrapper for reading the contents of a ZIP file.
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// fn doit() -> std::io::IoResult<()>
|
||||||
|
/// {
|
||||||
|
/// // For demonstration purposes we read from an empty buffer.
|
||||||
|
/// // Normally a File object would be used.
|
||||||
|
/// let buf = [0u8, 128];
|
||||||
|
/// let mut reader = std::io::BufReader::new(&buf);
|
||||||
|
///
|
||||||
|
/// let zip = try!(zip::ZipReader::new(reader));
|
||||||
|
///
|
||||||
|
/// for file in zip.files()
|
||||||
|
/// {
|
||||||
|
/// println!("Filename: {}", file.file_name_string());
|
||||||
|
/// let mut file_reader = try!(zip.read_file(file));
|
||||||
|
/// let first_byte = try!(file_reader.read_byte());
|
||||||
|
/// println!("{}", first_byte);
|
||||||
|
/// }
|
||||||
|
/// Ok(())
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// println!("Result: {}", doit());
|
||||||
|
/// ```
|
||||||
pub struct ZipReader<T>
|
pub struct ZipReader<T>
|
||||||
{
|
{
|
||||||
inner: RefCell<T>,
|
inner: RefCell<T>,
|
||||||
|
@ -25,6 +49,7 @@ fn unsupported_zip_error<T>(detail: &str) -> IoResult<T>
|
||||||
|
|
||||||
impl<T: Reader+Seek> ZipReader<T>
|
impl<T: Reader+Seek> ZipReader<T>
|
||||||
{
|
{
|
||||||
|
/// Opens a ZIP file and parses the content headers.
|
||||||
pub fn new(mut reader: T) -> IoResult<ZipReader<T>>
|
pub fn new(mut reader: T) -> IoResult<ZipReader<T>>
|
||||||
{
|
{
|
||||||
let footer = try!(spec::CentralDirectoryEnd::find_and_parse(&mut reader));
|
let footer = try!(spec::CentralDirectoryEnd::find_and_parse(&mut reader));
|
||||||
|
@ -45,11 +70,18 @@ impl<T: Reader+Seek> ZipReader<T>
|
||||||
Ok(ZipReader { inner: RefCell::new(reader), files: files })
|
Ok(ZipReader { inner: RefCell::new(reader), files: files })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// An iterator over the information of all contained files.
|
||||||
pub fn files(&self) -> ::std::slice::Items<ZipFile>
|
pub fn files(&self) -> ::std::slice::Items<ZipFile>
|
||||||
{
|
{
|
||||||
self.files.as_slice().iter()
|
self.files.as_slice().iter()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Gets a reader for a contained zipfile.
|
||||||
|
///
|
||||||
|
/// Possible errors:
|
||||||
|
///
|
||||||
|
/// * `ResourceUnavailable`: when another reader returned from this function is still active
|
||||||
|
/// * `OtherIoError`: if the file is encrypted or has an unsupported compression
|
||||||
pub fn read_file(&self, file: &ZipFile) -> IoResult<Box<Reader>>
|
pub fn read_file(&self, file: &ZipFile) -> IoResult<Box<Reader>>
|
||||||
{
|
{
|
||||||
let mut inner_reader = match self.inner.try_borrow_mut()
|
let mut inner_reader = match self.inner.try_borrow_mut()
|
||||||
|
|
34
src/types.rs
34
src/types.rs
|
@ -1,49 +1,81 @@
|
||||||
|
//! Types that specify what is contained in a ZIP.
|
||||||
|
|
||||||
use time;
|
use time;
|
||||||
|
|
||||||
|
/// Compression methods for the contents of a ZIP file.
|
||||||
#[deriving(FromPrimitive, Clone)]
|
#[deriving(FromPrimitive, Clone)]
|
||||||
pub enum CompressionMethod
|
pub enum CompressionMethod
|
||||||
{
|
{
|
||||||
|
/// The file is stored (no compression)
|
||||||
Stored = 0,
|
Stored = 0,
|
||||||
|
/// The file is Shrunk
|
||||||
Shrunk = 1,
|
Shrunk = 1,
|
||||||
|
/// The file is Reduced with compression factor 1
|
||||||
Reduced1 = 2,
|
Reduced1 = 2,
|
||||||
|
/// The file is Reduced with compression factor 2
|
||||||
Reduced2 = 3,
|
Reduced2 = 3,
|
||||||
|
/// The file is Reduced with compression factor 3
|
||||||
Reduced3 = 4,
|
Reduced3 = 4,
|
||||||
|
/// The file is Reduced with compression factor 4
|
||||||
Reduced4 = 5,
|
Reduced4 = 5,
|
||||||
|
/// The file is Imploded
|
||||||
Imploded = 6,
|
Imploded = 6,
|
||||||
|
/// The file is Deflated
|
||||||
Deflated = 8,
|
Deflated = 8,
|
||||||
|
/// Enhanced Deflating using Deflate64(tm)
|
||||||
Deflate64 = 9,
|
Deflate64 = 9,
|
||||||
|
/// PKWARE Data Compression Library Imploding (old IBM TERSE)
|
||||||
PkwareImploding = 10,
|
PkwareImploding = 10,
|
||||||
|
/// File is compressed using BZIP2 algorithm
|
||||||
Bzip2 = 12,
|
Bzip2 = 12,
|
||||||
|
/// LZMA (EFS)
|
||||||
LZMA = 14,
|
LZMA = 14,
|
||||||
|
/// File is compressed using IBM TERSE (new)
|
||||||
IBMTerse = 18,
|
IBMTerse = 18,
|
||||||
|
/// IBM LZ77 z Architecture (PFS)
|
||||||
LZ77 = 19,
|
LZ77 = 19,
|
||||||
|
/// WavPack compressed data
|
||||||
WavPack = 97,
|
WavPack = 97,
|
||||||
|
/// PPMd version I, Rev 1
|
||||||
PPMdI1 = 98,
|
PPMdI1 = 98,
|
||||||
|
/// Unknown (invalid) compression
|
||||||
Unknown = 100000,
|
Unknown = 100000,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Structure representing a ZIP file.
|
||||||
pub struct ZipFile
|
pub struct ZipFile
|
||||||
{
|
{
|
||||||
|
/// True if the file is encrypted.
|
||||||
pub encrypted: bool,
|
pub encrypted: bool,
|
||||||
|
/// Compression method used to store the file
|
||||||
pub compression_method: CompressionMethod,
|
pub compression_method: CompressionMethod,
|
||||||
|
/// Last modified time. This will only have a 2 second precision.
|
||||||
pub last_modified_time: time::Tm,
|
pub last_modified_time: time::Tm,
|
||||||
|
/// CRC32 checksum
|
||||||
pub crc32: u32,
|
pub crc32: u32,
|
||||||
|
/// Size of the file in the ZIP
|
||||||
pub compressed_size: u64,
|
pub compressed_size: u64,
|
||||||
|
/// Size of the file when extracted
|
||||||
pub uncompressed_size: u64,
|
pub uncompressed_size: u64,
|
||||||
|
/// Name of the file
|
||||||
pub file_name: Vec<u8>,
|
pub file_name: Vec<u8>,
|
||||||
|
/// File comment
|
||||||
pub file_comment: Vec<u8>,
|
pub file_comment: Vec<u8>,
|
||||||
|
/// Specifies where the local header of the file starts
|
||||||
pub header_start: u64,
|
pub header_start: u64,
|
||||||
|
/// Specifies where the compressed data of the file starts
|
||||||
pub data_start: u64,
|
pub data_start: u64,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ZipFile
|
impl ZipFile
|
||||||
{
|
{
|
||||||
|
/// Lossy UTF-8 interpretation of the file name
|
||||||
pub fn file_name_string(&self) -> String
|
pub fn file_name_string(&self) -> String
|
||||||
{
|
{
|
||||||
String::from_utf8_lossy(self.file_name.as_slice()).into_string()
|
String::from_utf8_lossy(self.file_name.as_slice()).into_string()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Lossy UTF-8 interpretation of the file comment
|
||||||
pub fn file_comment_string(&self) -> String
|
pub fn file_comment_string(&self) -> String
|
||||||
{
|
{
|
||||||
String::from_utf8_lossy(self.file_comment.as_slice()).into_string()
|
String::from_utf8_lossy(self.file_comment.as_slice()).into_string()
|
||||||
|
|
|
@ -17,6 +17,27 @@ enum GenericZipWriter<W>
|
||||||
Deflater(DeflateEncoder<W>),
|
Deflater(DeflateEncoder<W>),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Generator for ZIP files.
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// fn doit() -> std::io::IoResult<()>
|
||||||
|
/// {
|
||||||
|
/// // For this example we write to a buffer, but normally you should use a File
|
||||||
|
/// let mut buf = [0u8, ..65536];
|
||||||
|
/// let w = std::io::BufWriter::new(&mut buf);
|
||||||
|
/// let mut zip = zip::ZipWriter::new(w);
|
||||||
|
///
|
||||||
|
/// try!(zip.start_file(b"hello_world.txt", zip::types::Stored));
|
||||||
|
/// try!(zip.write(b"Hello, World!"));
|
||||||
|
///
|
||||||
|
/// // Optionally finish the zip. (this is also done on drop)
|
||||||
|
/// try!(zip.finalize());
|
||||||
|
///
|
||||||
|
/// Ok(())
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// println!("Result: {}", doit());
|
||||||
|
/// ```
|
||||||
pub struct ZipWriter<W>
|
pub struct ZipWriter<W>
|
||||||
{
|
{
|
||||||
inner: GenericZipWriter<W>,
|
inner: GenericZipWriter<W>,
|
||||||
|
@ -63,6 +84,9 @@ impl ZipWriterStats
|
||||||
|
|
||||||
impl<W: Writer+Seek> ZipWriter<W>
|
impl<W: Writer+Seek> ZipWriter<W>
|
||||||
{
|
{
|
||||||
|
/// Initializes the ZipWriter.
|
||||||
|
///
|
||||||
|
/// Before writing to this object, the start_file command should be called.
|
||||||
pub fn new(inner: W) -> ZipWriter<W>
|
pub fn new(inner: W) -> ZipWriter<W>
|
||||||
{
|
{
|
||||||
ZipWriter
|
ZipWriter
|
||||||
|
@ -73,6 +97,7 @@ impl<W: Writer+Seek> ZipWriter<W>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Start a new file for with the requested compression method.
|
||||||
pub fn start_file(&mut self, name: &[u8], compression: types::CompressionMethod) -> IoResult<()>
|
pub fn start_file(&mut self, name: &[u8], compression: types::CompressionMethod) -> IoResult<()>
|
||||||
{
|
{
|
||||||
try!(self.finish_file());
|
try!(self.finish_file());
|
||||||
|
@ -126,6 +151,7 @@ impl<W: Writer+Seek> ZipWriter<W>
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Finish the last file and write all other zip-structures
|
||||||
pub fn finalize(&mut self) -> IoResult<()>
|
pub fn finalize(&mut self) -> IoResult<()>
|
||||||
{
|
{
|
||||||
try!(self.finish_file());
|
try!(self.finish_file());
|
||||||
|
|
Loading…
Add table
Reference in a new issue