From 0eea88d6c033aa9d79a47d3329d9f974c499b692 Mon Sep 17 00:00:00 2001 From: Marli Frost Date: Sun, 23 Jan 2022 21:43:28 +0000 Subject: [PATCH 1/4] doc: more beginner-friendly DateTime give more warnings about possible misuse of the type --- src/types.rs | 47 ++++++++++++++++++++++++++++++++++++++--------- 1 file changed, 38 insertions(+), 9 deletions(-) diff --git a/src/types.rs b/src/types.rs index 88434e3f..523de6a1 100644 --- a/src/types.rs +++ b/src/types.rs @@ -1,4 +1,9 @@ //! Types that specify what is contained in a ZIP. +#[cfg(doc)] +use { + crate::read::ZipFile, + crate::write::FileOptions, +}; #[cfg(feature = "time")] use time::{error::ComponentRange, Date, Month, OffsetDateTime, PrimitiveDateTime, Time}; @@ -22,19 +27,23 @@ impl System { } } -/// A DateTime field to be used for storing timestamps in a zip file +/// Representation of a moment in time. +/// +/// Zip files use an old format from DOS to store timestamps, +/// with its own set of peculiarities. +/// For example, it has a resolution of 2 seconds! /// -/// This structure does bounds checking to ensure the date is able to be stored in a zip file. -/// -/// When constructed manually from a date and time, it will also check if the input is sensible -/// (e.g. months are from [1, 12]), but when read from a zip some parts may be out of their normal -/// bounds (e.g. month 0, or hour 31). +/// A [`DateTime`] can be stored directly in a zipfile with [`FileOptions::last_modified_time`], +/// or read from one with [`ZipFile::last_modified`] /// /// # Warning +/// +/// Because there is no timezone associated with the [`DateTime`], they should ideally only +/// be used for user-facing descriptions. This also means [`DateTime::to_time`] returns an +/// [`OffsetDateTime`] (which is the equivalent of chrono's `NaiveDateTime`). /// -/// Some utilities use alternative timestamps to improve the accuracy of their -/// ZIPs, but we don't parse them yet. [We're working on this](https://github.com/zip-rs/zip/issues/156#issuecomment-652981904), -/// however this API shouldn't be considered complete. +/// Modern zip files store more precise timestamps, which are ignored by [`crate::read::ZipArchive`], +/// so keep in mind that these timestamps are unreliable. [We're working on this](https://github.com/zip-rs/zip/issues/156#issuecomment-652981904). #[derive(Debug, Clone, Copy)] pub struct DateTime { year: u16, @@ -166,26 +175,46 @@ impl DateTime { } /// Get the month, where 1 = january and 12 = december + /// + /// # Warning + /// + /// When read from a zip file, this may not be a reasonable value pub fn month(&self) -> u8 { self.month } /// Get the day + /// + /// # Warning + /// + /// When read from a zip file, this may not be a reasonable value pub fn day(&self) -> u8 { self.day } /// Get the hour + /// + /// # Warning + /// + /// When read from a zip file, this may not be a reasonable value pub fn hour(&self) -> u8 { self.hour } /// Get the minute + /// + /// # Warning + /// + /// When read from a zip file, this may not be a reasonable value pub fn minute(&self) -> u8 { self.minute } /// Get the second + /// + /// # Warning + /// + /// When read from a zip file, this may not be a reasonable value pub fn second(&self) -> u8 { self.second } From f1074bc6a9928eb9be549eef04d25c9411619695 Mon Sep 17 00:00:00 2001 From: Marli Frost Date: Sun, 23 Jan 2022 21:45:41 +0000 Subject: [PATCH 2/4] doc: remove re-exports section from crate root Making the paths to the types private forces rustdoc to render the structs inline in the crate root. This is simpler to see when first reading the API doc --- src/read.rs | 52 ++++++++++++++++++---------------- src/write.rs | 80 +++++++++++++++++++++++++++------------------------- 2 files changed, 70 insertions(+), 62 deletions(-) diff --git a/src/read.rs b/src/read.rs index a9f173fd..4dff725b 100644 --- a/src/read.rs +++ b/src/read.rs @@ -32,31 +32,35 @@ mod ffi { pub const S_IFREG: u32 = 0o0100000; } -/// ZIP archive reader -/// -/// ```no_run -/// use std::io::prelude::*; -/// fn list_zip_contents(reader: impl Read + Seek) -> zip::result::ZipResult<()> { -/// let mut zip = zip::ZipArchive::new(reader)?; -/// -/// for i in 0..zip.len() { -/// let mut file = zip.by_index(i)?; -/// println!("Filename: {}", file.name()); -/// std::io::copy(&mut file, &mut std::io::stdout()); -/// } -/// -/// Ok(()) -/// } -/// ``` -#[derive(Clone, Debug)] -pub struct ZipArchive { - reader: R, - files: Vec, - names_map: HashMap, - offset: u64, - comment: Vec, -} +// Put the struct declaration in a private module to convince rustdoc to display ZipArchive nicely +pub(crate) mod zip_archive { + /// ZIP archive reader + /// + /// ```no_run + /// use std::io::prelude::*; + /// fn list_zip_contents(reader: impl Read + Seek) -> zip::result::ZipResult<()> { + /// let mut zip = zip::ZipArchive::new(reader)?; + /// + /// for i in 0..zip.len() { + /// let mut file = zip.by_index(i)?; + /// println!("Filename: {}", file.name()); + /// std::io::copy(&mut file, &mut std::io::stdout()); + /// } + /// + /// Ok(()) + /// } + /// ``` + #[derive(Clone, Debug)] + pub struct ZipArchive { + pub(super) reader: R, + pub(super) files: Vec, + pub(super) names_map: super::HashMap, + pub(super) offset: u64, + pub(super) comment: Vec, + } +} +pub use zip_archive::ZipArchive; enum CryptoReader<'a> { Plaintext(io::Take<&'a mut dyn Read>), ZipCrypto(ZipCryptoReaderValid>), diff --git a/src/write.rs b/src/write.rs index 17fd3344..7c9ccdf3 100644 --- a/src/write.rs +++ b/src/write.rs @@ -42,45 +42,49 @@ enum GenericZipWriter { #[cfg(feature = "zstd")] Zstd(ZstdEncoder<'static, W>), } - -/// ZIP archive generator -/// -/// Handles the bookkeeping involved in building an archive, and provides an -/// API to edit its contents. -/// -/// ``` -/// # fn doit() -> zip::result::ZipResult<()> -/// # { -/// # use zip::ZipWriter; -/// use std::io::Write; -/// use zip::write::FileOptions; -/// -/// // We use a buffer here, though you'd normally use a `File` -/// let mut buf = [0; 65536]; -/// let mut zip = zip::ZipWriter::new(std::io::Cursor::new(&mut buf[..])); -/// -/// let options = zip::write::FileOptions::default().compression_method(zip::CompressionMethod::Stored); -/// zip.start_file("hello_world.txt", options)?; -/// zip.write(b"Hello, World!")?; -/// -/// // Apply the changes you've made. -/// // Dropping the `ZipWriter` will have the same effect, but may silently fail -/// zip.finish()?; -/// -/// # Ok(()) -/// # } -/// # doit().unwrap(); -/// ``` -pub struct ZipWriter { - inner: GenericZipWriter, - files: Vec, - stats: ZipWriterStats, - writing_to_file: bool, - writing_to_extra_field: bool, - writing_to_central_extra_field_only: bool, - writing_raw: bool, - comment: Vec, +// Put the struct declaration in a private module to convince rustdoc to display ZipWriter nicely +pub(crate) mod zip_writer { + use super::*; + /// ZIP archive generator + /// + /// Handles the bookkeeping involved in building an archive, and provides an + /// API to edit its contents. + /// + /// ``` + /// # fn doit() -> zip::result::ZipResult<()> + /// # { + /// # use zip::ZipWriter; + /// use std::io::Write; + /// use zip::write::FileOptions; + /// + /// // We use a buffer here, though you'd normally use a `File` + /// let mut buf = [0; 65536]; + /// let mut zip = zip::ZipWriter::new(std::io::Cursor::new(&mut buf[..])); + /// + /// let options = zip::write::FileOptions::default().compression_method(zip::CompressionMethod::Stored); + /// zip.start_file("hello_world.txt", options)?; + /// zip.write(b"Hello, World!")?; + /// + /// // Apply the changes you've made. + /// // Dropping the `ZipWriter` will have the same effect, but may silently fail + /// zip.finish()?; + /// + /// # Ok(()) + /// # } + /// # doit().unwrap(); + /// ``` + pub struct ZipWriter { + pub(super) inner: GenericZipWriter, + pub(super) files: Vec, + pub(super) stats: ZipWriterStats, + pub(super) writing_to_file: bool, + pub(super) writing_to_extra_field: bool, + pub(super) writing_to_central_extra_field_only: bool, + pub(super) writing_raw: bool, + pub(super) comment: Vec, + } } +pub use zip_writer::ZipWriter; #[derive(Default)] struct ZipWriterStats { From ae941ad2566f00e2fadf82bf9c4017055e3abd51 Mon Sep 17 00:00:00 2001 From: Marli Frost Date: Sun, 23 Jan 2022 21:46:17 +0000 Subject: [PATCH 3/4] doc: improve landing page Explain more of what ZIP is intended for, and begin to explain what the crate can be used for. --- src/lib.rs | 23 +++++++++++++++++++++-- 1 file changed, 21 insertions(+), 2 deletions(-) diff --git a/src/lib.rs b/src/lib.rs index 3b39ab4f..8a20d374 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -1,7 +1,26 @@ -//! An ergonomic API for reading and writing ZIP files. +//! A library for reading and writing ZIP archives. +//! ZIP is a format designed for cross-platform file "archiving". +//! That is, storing a collection of files in a single datastream +//! to make them easier to share between computers. +//! Additionally, ZIP is able to compress and encrypt files in its +//! archives. //! //! The current implementation is based on [PKWARE's APPNOTE.TXT v6.3.9](https://pkware.cachefly.net/webdocs/casestudies/APPNOTE.TXT) -// TODO(#184): Decide on the crate's bias: Do we prioritise permissiveness/correctness/speed/ergonomics? +//! +//! --- +//! +//! [`zip`](`crate`) has support for the most common ZIP archives found in common use. +//! However, in special cases, +//! there are some zip archives that are difficult to read or write. +//! +//! This is a list of supported features: +//! +//! | | Reading | Writing | +//! | ------- | ------ | ------- | +//! | Deflate | ✅ [->](`crate::ZipArchive::by_name`) | ✅ [->](`crate::write::FileOptions::compression_method`) | +//! +//! +//! #![warn(missing_docs)] From b080731c5543043728137cf2f5ac75d538da8ff4 Mon Sep 17 00:00:00 2001 From: Marli Frost Date: Sun, 23 Jan 2022 21:53:42 +0000 Subject: [PATCH 4/4] chore: rustfmt --- src/lib.rs | 14 +++++++------- src/types.rs | 29 +++++++++++++---------------- 2 files changed, 20 insertions(+), 23 deletions(-) diff --git a/src/lib.rs b/src/lib.rs index 8a20d374..24e4377c 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -6,21 +6,21 @@ //! archives. //! //! The current implementation is based on [PKWARE's APPNOTE.TXT v6.3.9](https://pkware.cachefly.net/webdocs/casestudies/APPNOTE.TXT) -//! +//! //! --- -//! +//! //! [`zip`](`crate`) has support for the most common ZIP archives found in common use. //! However, in special cases, //! there are some zip archives that are difficult to read or write. -//! +//! //! This is a list of supported features: -//! +//! //! | | Reading | Writing | //! | ------- | ------ | ------- | //! | Deflate | ✅ [->](`crate::ZipArchive::by_name`) | ✅ [->](`crate::write::FileOptions::compression_method`) | -//! -//! -//! +//! +//! +//! #![warn(missing_docs)] diff --git a/src/types.rs b/src/types.rs index 523de6a1..e68c934e 100644 --- a/src/types.rs +++ b/src/types.rs @@ -1,9 +1,6 @@ //! Types that specify what is contained in a ZIP. #[cfg(doc)] -use { - crate::read::ZipFile, - crate::write::FileOptions, -}; +use {crate::read::ZipFile, crate::write::FileOptions}; #[cfg(feature = "time")] use time::{error::ComponentRange, Date, Month, OffsetDateTime, PrimitiveDateTime, Time}; @@ -28,7 +25,7 @@ impl System { } /// Representation of a moment in time. -/// +/// /// Zip files use an old format from DOS to store timestamps, /// with its own set of peculiarities. /// For example, it has a resolution of 2 seconds! @@ -37,7 +34,7 @@ impl System { /// or read from one with [`ZipFile::last_modified`] /// /// # Warning -/// +/// /// Because there is no timezone associated with the [`DateTime`], they should ideally only /// be used for user-facing descriptions. This also means [`DateTime::to_time`] returns an /// [`OffsetDateTime`] (which is the equivalent of chrono's `NaiveDateTime`). @@ -175,45 +172,45 @@ impl DateTime { } /// Get the month, where 1 = january and 12 = december - /// + /// /// # Warning - /// + /// /// When read from a zip file, this may not be a reasonable value pub fn month(&self) -> u8 { self.month } /// Get the day - /// + /// /// # Warning - /// + /// /// When read from a zip file, this may not be a reasonable value pub fn day(&self) -> u8 { self.day } /// Get the hour - /// + /// /// # Warning - /// + /// /// When read from a zip file, this may not be a reasonable value pub fn hour(&self) -> u8 { self.hour } /// Get the minute - /// + /// /// # Warning - /// + /// /// When read from a zip file, this may not be a reasonable value pub fn minute(&self) -> u8 { self.minute } /// Get the second - /// + /// /// # Warning - /// + /// /// When read from a zip file, this may not be a reasonable value pub fn second(&self) -> u8 { self.second