Added documentation and moved types to top

This commit is contained in:
Mathijs van de Nes 2014-09-11 11:48:44 +02:00
parent 4d07f695de
commit 02059ce351
7 changed files with 106 additions and 9 deletions

View file

@ -6,7 +6,7 @@ fn main()
let fname = Path::new(args[1].as_slice());
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()
{

View file

@ -11,11 +11,7 @@ fn doit() -> std::io::IoResult<()>
let fname = Path::new(args[1].as_slice());
let file = std::io::File::create(&fname).unwrap();
let mut zip = zip::writer::ZipWriter::new(file);
// try!(zip.start_file(b"test", zip::types::Stored));
// try!(zip.write(b""));
let mut zip = zip::ZipWriter::new(file);
try!(zip.start_file(b"test/readme.txt", zip::types::Stored));
try!(zip.write(b"Hello, World!\n"));

View file

@ -1,3 +1,5 @@
//! Helper module to compute a CRC32 checksum
use std::io;
static CRC32_TABLE : [u32, ..256] = [
@ -46,6 +48,7 @@ static CRC32_TABLE : [u32, ..256] = [
0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, 0x2d02ef8d
];
/// Update the checksum original based upon the contents of buf.
pub fn crc32(original: u32, buf: &[u8]) -> u32
{
let mut crc = original ^ !0u32;
@ -58,6 +61,7 @@ pub fn crc32(original: u32, buf: &[u8]) -> u32
return crc ^ !0u32;
}
/// Reader that validates the CRC32 when it reaches the EOF.
pub struct Crc32Reader<R>
{
inner: R,
@ -67,6 +71,7 @@ pub struct 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>
{
Crc32Reader

View file

@ -1,13 +1,19 @@
//! A basic ZipReader/Writer crate
#![feature(phase)]
#![feature(unsafe_destructor)]
#![warn(missing_doc)]
#[phase(plugin, link)] extern crate log;
extern crate time;
extern crate flate2;
pub use reader::ZipReader;
pub use writer::ZipWriter;
mod util;
mod spec;
pub mod crc32;
pub mod reader;
mod reader;
pub mod types;
pub mod writer;
mod writer;

View file

@ -7,6 +7,30 @@ use std::io::{IoResult, IoError};
use std::cell::RefCell;
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>
{
inner: RefCell<T>,
@ -25,6 +49,7 @@ fn unsupported_zip_error<T>(detail: &str) -> IoResult<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>>
{
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 })
}
/// An iterator over the information of all contained files.
pub fn files(&self) -> ::std::slice::Items<ZipFile>
{
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>>
{
let mut inner_reader = match self.inner.try_borrow_mut()

View file

@ -1,49 +1,81 @@
//! Types that specify what is contained in a ZIP.
use time;
/// Compression methods for the contents of a ZIP file.
#[deriving(FromPrimitive, Clone)]
pub enum CompressionMethod
{
/// The file is stored (no compression)
Stored = 0,
/// The file is Shrunk
Shrunk = 1,
/// The file is Reduced with compression factor 1
Reduced1 = 2,
/// The file is Reduced with compression factor 2
Reduced2 = 3,
/// The file is Reduced with compression factor 3
Reduced3 = 4,
/// The file is Reduced with compression factor 4
Reduced4 = 5,
/// The file is Imploded
Imploded = 6,
/// The file is Deflated
Deflated = 8,
/// Enhanced Deflating using Deflate64(tm)
Deflate64 = 9,
/// PKWARE Data Compression Library Imploding (old IBM TERSE)
PkwareImploding = 10,
/// File is compressed using BZIP2 algorithm
Bzip2 = 12,
/// LZMA (EFS)
LZMA = 14,
/// File is compressed using IBM TERSE (new)
IBMTerse = 18,
/// IBM LZ77 z Architecture (PFS)
LZ77 = 19,
/// WavPack compressed data
WavPack = 97,
/// PPMd version I, Rev 1
PPMdI1 = 98,
/// Unknown (invalid) compression
Unknown = 100000,
}
/// Structure representing a ZIP file.
pub struct ZipFile
{
/// True if the file is encrypted.
pub encrypted: bool,
/// Compression method used to store the file
pub compression_method: CompressionMethod,
/// Last modified time. This will only have a 2 second precision.
pub last_modified_time: time::Tm,
/// CRC32 checksum
pub crc32: u32,
/// Size of the file in the ZIP
pub compressed_size: u64,
/// Size of the file when extracted
pub uncompressed_size: u64,
/// Name of the file
pub file_name: Vec<u8>,
/// File comment
pub file_comment: Vec<u8>,
/// Specifies where the local header of the file starts
pub header_start: u64,
/// Specifies where the compressed data of the file starts
pub data_start: u64,
}
impl ZipFile
{
/// Lossy UTF-8 interpretation of the file name
pub fn file_name_string(&self) -> 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
{
String::from_utf8_lossy(self.file_comment.as_slice()).into_string()

View file

@ -17,6 +17,27 @@ enum GenericZipWriter<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>
{
inner: GenericZipWriter<W>,
@ -63,6 +84,9 @@ impl ZipWriterStats
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>
{
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<()>
{
try!(self.finish_file());
@ -126,6 +151,7 @@ impl<W: Writer+Seek> ZipWriter<W>
Ok(())
}
/// Finish the last file and write all other zip-structures
pub fn finalize(&mut self) -> IoResult<()>
{
try!(self.finish_file());