Change Rc to Arc since sharing across ZipWriter instances is possible
This commit is contained in:
parent
651a87f881
commit
df3dfc1991
4 changed files with 41 additions and 37 deletions
|
@ -142,4 +142,8 @@
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- Fixes a rare bug where the size of the extra-data field could overflow when `large_file` was set.
|
- Fixes a rare bug where the size of the extra-data field could overflow when `large_file` was set.
|
||||||
- Fixes more cases of a bug when ZIP64 magic bytes occur in filenames.
|
- Fixes more cases of a bug when ZIP64 magic bytes occur in filenames.
|
||||||
|
|
||||||
|
## [0.8.1]
|
||||||
|
|
||||||
|
- `ZipWriter` now implements `Send` if the underlying writer does.
|
23
src/read.rs
23
src/read.rs
|
@ -14,7 +14,6 @@ use std::borrow::Cow;
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use std::io::{self, prelude::*};
|
use std::io::{self, prelude::*};
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
use std::rc::Rc;
|
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
#[cfg(any(
|
#[cfg(any(
|
||||||
|
@ -551,7 +550,7 @@ impl<R: Read + Seek> ZipArchive<R> {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Search for a file entry by name
|
/// Search for a file entry by name
|
||||||
pub fn by_name<'a>(&'a mut self, name: &str) -> ZipResult<ZipFile<'a>> {
|
pub fn by_name(&mut self, name: &str) -> ZipResult<ZipFile> {
|
||||||
Ok(self.by_name_with_optional_password(name, None)?.unwrap())
|
Ok(self.by_name_with_optional_password(name, None)?.unwrap())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -582,11 +581,11 @@ impl<R: Read + Seek> ZipArchive<R> {
|
||||||
/// There are many passwords out there that will also pass the validity checks
|
/// There are many passwords out there that will also pass the validity checks
|
||||||
/// we are able to perform. This is a weakness of the ZipCrypto algorithm,
|
/// we are able to perform. This is a weakness of the ZipCrypto algorithm,
|
||||||
/// due to its fairly primitive approach to cryptography.
|
/// due to its fairly primitive approach to cryptography.
|
||||||
pub fn by_index_decrypt<'a>(
|
pub fn by_index_decrypt(
|
||||||
&'a mut self,
|
&mut self,
|
||||||
file_number: usize,
|
file_number: usize,
|
||||||
password: &[u8],
|
password: &[u8],
|
||||||
) -> ZipResult<Result<ZipFile<'a>, InvalidPassword>> {
|
) -> ZipResult<Result<ZipFile, InvalidPassword>> {
|
||||||
self.by_index_with_optional_password(file_number, Some(password))
|
self.by_index_with_optional_password(file_number, Some(password))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -613,11 +612,11 @@ impl<R: Read + Seek> ZipArchive<R> {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn by_index_with_optional_password<'a>(
|
fn by_index_with_optional_password(
|
||||||
&'a mut self,
|
&mut self,
|
||||||
file_number: usize,
|
file_number: usize,
|
||||||
mut password: Option<&[u8]>,
|
mut password: Option<&[u8]>,
|
||||||
) -> ZipResult<Result<ZipFile<'a>, InvalidPassword>> {
|
) -> ZipResult<Result<ZipFile, InvalidPassword>> {
|
||||||
let data = self
|
let data = self
|
||||||
.shared
|
.shared
|
||||||
.files
|
.files
|
||||||
|
@ -738,8 +737,8 @@ fn central_header_to_zip_file_inner<R: Read>(
|
||||||
uncompressed_size: uncompressed_size as u64,
|
uncompressed_size: uncompressed_size as u64,
|
||||||
file_name,
|
file_name,
|
||||||
file_name_raw,
|
file_name_raw,
|
||||||
extra_field: Rc::new(extra_field),
|
extra_field: Arc::new(extra_field),
|
||||||
central_extra_field: Rc::new(vec![]),
|
central_extra_field: Arc::new(vec![]),
|
||||||
file_comment,
|
file_comment,
|
||||||
header_start: offset,
|
header_start: offset,
|
||||||
central_header_start,
|
central_header_start,
|
||||||
|
@ -1099,8 +1098,8 @@ pub fn read_zipfile_from_stream<'a, R: Read>(reader: &'a mut R) -> ZipResult<Opt
|
||||||
uncompressed_size: uncompressed_size as u64,
|
uncompressed_size: uncompressed_size as u64,
|
||||||
file_name,
|
file_name,
|
||||||
file_name_raw,
|
file_name_raw,
|
||||||
extra_field: Rc::new(extra_field),
|
extra_field: Arc::new(extra_field),
|
||||||
central_extra_field: Rc::new(vec![]),
|
central_extra_field: Arc::new(vec![]),
|
||||||
file_comment: String::new(), // file comment is only available in the central directory
|
file_comment: String::new(), // file comment is only available in the central directory
|
||||||
// header_start and data start are not available, but also don't matter, since seeking is
|
// header_start and data start are not available, but also don't matter, since seeking is
|
||||||
// not available.
|
// not available.
|
||||||
|
|
11
src/types.rs
11
src/types.rs
|
@ -1,7 +1,7 @@
|
||||||
//! Types that specify what is contained in a ZIP.
|
//! Types that specify what is contained in a ZIP.
|
||||||
use path::{Component, Path, PathBuf};
|
use path::{Component, Path, PathBuf};
|
||||||
use std::path;
|
use std::path;
|
||||||
use std::rc::Rc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
#[cfg(not(any(
|
#[cfg(not(any(
|
||||||
all(target_arch = "arm", target_pointer_width = "32"),
|
all(target_arch = "arm", target_pointer_width = "32"),
|
||||||
|
@ -362,9 +362,9 @@ pub struct ZipFileData {
|
||||||
/// Raw file name. To be used when file_name was incorrectly decoded.
|
/// Raw file name. To be used when file_name was incorrectly decoded.
|
||||||
pub file_name_raw: Vec<u8>,
|
pub file_name_raw: Vec<u8>,
|
||||||
/// Extra field usually used for storage expansion
|
/// Extra field usually used for storage expansion
|
||||||
pub extra_field: Rc<Vec<u8>>,
|
pub extra_field: Arc<Vec<u8>>,
|
||||||
/// Extra field only written to central directory
|
/// Extra field only written to central directory
|
||||||
pub central_extra_field: Rc<Vec<u8>>,
|
pub central_extra_field: Arc<Vec<u8>>,
|
||||||
/// File comment
|
/// File comment
|
||||||
pub file_comment: String,
|
pub file_comment: String,
|
||||||
/// Specifies where the local header of the file starts
|
/// Specifies where the local header of the file starts
|
||||||
|
@ -531,8 +531,8 @@ mod test {
|
||||||
uncompressed_size: 0,
|
uncompressed_size: 0,
|
||||||
file_name: file_name.clone(),
|
file_name: file_name.clone(),
|
||||||
file_name_raw: file_name.into_bytes(),
|
file_name_raw: file_name.into_bytes(),
|
||||||
extra_field: Rc::new(vec![]),
|
extra_field: Arc::new(vec![]),
|
||||||
central_extra_field: Rc::new(vec![]),
|
central_extra_field: Arc::new(vec![]),
|
||||||
file_comment: String::new(),
|
file_comment: String::new(),
|
||||||
header_start: 0,
|
header_start: 0,
|
||||||
data_start: AtomicU64::new(0),
|
data_start: AtomicU64::new(0),
|
||||||
|
@ -581,6 +581,7 @@ mod test {
|
||||||
assert!(DateTime::from_date_and_time(2107, 12, 32, 0, 0, 0).is_err());
|
assert!(DateTime::from_date_and_time(2107, 12, 32, 0, 0, 0).is_err());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
use std::sync::Arc;
|
||||||
#[cfg(feature = "time")]
|
#[cfg(feature = "time")]
|
||||||
use time::{format_description::well_known::Rfc3339, OffsetDateTime};
|
use time::{format_description::well_known::Rfc3339, OffsetDateTime};
|
||||||
|
|
||||||
|
|
38
src/write.rs
38
src/write.rs
|
@ -14,7 +14,7 @@ use std::io;
|
||||||
use std::io::prelude::*;
|
use std::io::prelude::*;
|
||||||
use std::io::{BufReader, SeekFrom};
|
use std::io::{BufReader, SeekFrom};
|
||||||
use std::mem;
|
use std::mem;
|
||||||
use std::rc::Rc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
#[cfg(any(
|
#[cfg(any(
|
||||||
feature = "deflate",
|
feature = "deflate",
|
||||||
|
@ -134,8 +134,8 @@ pub struct FileOptions {
|
||||||
pub(crate) permissions: Option<u32>,
|
pub(crate) permissions: Option<u32>,
|
||||||
pub(crate) large_file: bool,
|
pub(crate) large_file: bool,
|
||||||
encrypt_with: Option<ZipCryptoKeys>,
|
encrypt_with: Option<ZipCryptoKeys>,
|
||||||
extra_data: Rc<Vec<u8>>,
|
extra_data: Arc<Vec<u8>>,
|
||||||
central_extra_data: Rc<Vec<u8>>,
|
central_extra_data: Arc<Vec<u8>>,
|
||||||
alignment: u16,
|
alignment: u16,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -149,8 +149,8 @@ impl arbitrary::Arbitrary<'_> for FileOptions {
|
||||||
permissions: Option::<u32>::arbitrary(u)?,
|
permissions: Option::<u32>::arbitrary(u)?,
|
||||||
large_file: bool::arbitrary(u)?,
|
large_file: bool::arbitrary(u)?,
|
||||||
encrypt_with: Option::<ZipCryptoKeys>::arbitrary(u)?,
|
encrypt_with: Option::<ZipCryptoKeys>::arbitrary(u)?,
|
||||||
extra_data: Rc::new(vec![]),
|
extra_data: Arc::new(vec![]),
|
||||||
central_extra_data: Rc::new(vec![]),
|
central_extra_data: Arc::new(vec![]),
|
||||||
alignment: u16::arbitrary(u)?,
|
alignment: u16::arbitrary(u)?,
|
||||||
};
|
};
|
||||||
u.arbitrary_loop(Some(0), Some((u16::MAX / 4) as u32), |u| {
|
u.arbitrary_loop(Some(0), Some((u16::MAX / 4) as u32), |u| {
|
||||||
|
@ -252,12 +252,12 @@ impl FileOptions {
|
||||||
} else {
|
} else {
|
||||||
&mut self.extra_data
|
&mut self.extra_data
|
||||||
};
|
};
|
||||||
let vec = Rc::get_mut(field);
|
let vec = Arc::get_mut(field);
|
||||||
let vec = match vec {
|
let vec = match vec {
|
||||||
Some(exclusive) => exclusive,
|
Some(exclusive) => exclusive,
|
||||||
None => {
|
None => {
|
||||||
*field = Rc::new(field.to_vec());
|
*field = Arc::new(field.to_vec());
|
||||||
Rc::get_mut(field).unwrap()
|
Arc::get_mut(field).unwrap()
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
vec.reserve_exact(data.len() + 4);
|
vec.reserve_exact(data.len() + 4);
|
||||||
|
@ -272,10 +272,10 @@ impl FileOptions {
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub fn clear_extra_data(mut self) -> FileOptions {
|
pub fn clear_extra_data(mut self) -> FileOptions {
|
||||||
if self.extra_data.len() > 0 {
|
if self.extra_data.len() > 0 {
|
||||||
self.extra_data = Rc::new(vec![]);
|
self.extra_data = Arc::new(vec![]);
|
||||||
}
|
}
|
||||||
if self.central_extra_data.len() > 0 {
|
if self.central_extra_data.len() > 0 {
|
||||||
self.central_extra_data = Rc::new(vec![]);
|
self.central_extra_data = Arc::new(vec![]);
|
||||||
}
|
}
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
@ -305,8 +305,8 @@ impl Default for FileOptions {
|
||||||
permissions: None,
|
permissions: None,
|
||||||
large_file: false,
|
large_file: false,
|
||||||
encrypt_with: None,
|
encrypt_with: None,
|
||||||
extra_data: Rc::new(vec![]),
|
extra_data: Arc::new(vec![]),
|
||||||
central_extra_data: Rc::new(vec![]),
|
central_extra_data: Arc::new(vec![]),
|
||||||
alignment: 1,
|
alignment: 1,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1398,7 +1398,7 @@ mod test {
|
||||||
use crate::ZipArchive;
|
use crate::ZipArchive;
|
||||||
use std::io;
|
use std::io;
|
||||||
use std::io::{Read, Write};
|
use std::io::{Read, Write};
|
||||||
use std::rc::Rc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn write_empty_zip() {
|
fn write_empty_zip() {
|
||||||
|
@ -1518,8 +1518,8 @@ mod test {
|
||||||
permissions: Some(33188),
|
permissions: Some(33188),
|
||||||
large_file: false,
|
large_file: false,
|
||||||
encrypt_with: None,
|
encrypt_with: None,
|
||||||
extra_data: Rc::new(vec![]),
|
extra_data: Arc::new(vec![]),
|
||||||
central_extra_data: Rc::new(vec![]),
|
central_extra_data: Arc::new(vec![]),
|
||||||
alignment: 1,
|
alignment: 1,
|
||||||
};
|
};
|
||||||
writer.start_file("mimetype", options).unwrap();
|
writer.start_file("mimetype", options).unwrap();
|
||||||
|
@ -1558,8 +1558,8 @@ mod test {
|
||||||
permissions: Some(33188),
|
permissions: Some(33188),
|
||||||
large_file: false,
|
large_file: false,
|
||||||
encrypt_with: None,
|
encrypt_with: None,
|
||||||
extra_data: Rc::new(vec![]),
|
extra_data: Arc::new(vec![]),
|
||||||
central_extra_data: Rc::new(vec![]),
|
central_extra_data: Arc::new(vec![]),
|
||||||
alignment: 0,
|
alignment: 0,
|
||||||
};
|
};
|
||||||
writer.start_file(RT_TEST_FILENAME, options).unwrap();
|
writer.start_file(RT_TEST_FILENAME, options).unwrap();
|
||||||
|
@ -1608,8 +1608,8 @@ mod test {
|
||||||
permissions: Some(33188),
|
permissions: Some(33188),
|
||||||
large_file: false,
|
large_file: false,
|
||||||
encrypt_with: None,
|
encrypt_with: None,
|
||||||
extra_data: Rc::new(vec![]),
|
extra_data: Arc::new(vec![]),
|
||||||
central_extra_data: Rc::new(vec![]),
|
central_extra_data: Arc::new(vec![]),
|
||||||
alignment: 0,
|
alignment: 0,
|
||||||
};
|
};
|
||||||
writer.start_file(RT_TEST_FILENAME, options).unwrap();
|
writer.start_file(RT_TEST_FILENAME, options).unwrap();
|
||||||
|
|
Loading…
Add table
Reference in a new issue