Merge remote-tracking branch 'origin/master' into feature/bump_msrv
This commit is contained in:
commit
4a50d24b4d
4 changed files with 40 additions and 90 deletions
|
@ -12,7 +12,7 @@ edition = "2018"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
flate2 = { version = "1.0.0", default-features = false, optional = true }
|
flate2 = { version = "1.0.0", default-features = false, optional = true }
|
||||||
time = { version = "0.1", optional = true }
|
time = { version = "0.3", features = ["formatting", "macros" ], optional = true }
|
||||||
byteorder = "1.3"
|
byteorder = "1.3"
|
||||||
bzip2 = { version = "0.4", optional = true }
|
bzip2 = { version = "0.4", optional = true }
|
||||||
crc32fast = "1.0"
|
crc32fast = "1.0"
|
||||||
|
|
|
@ -9,7 +9,7 @@ use std::fmt;
|
||||||
/// contents to be read without context.
|
/// contents to be read without context.
|
||||||
///
|
///
|
||||||
/// When creating ZIP files, you may choose the method to use with
|
/// When creating ZIP files, you may choose the method to use with
|
||||||
/// [`zip::write::FileOptions::compression_method`]
|
/// [`crate::write::FileOptions::compression_method`]
|
||||||
#[derive(Copy, Clone, PartialEq, Eq, Debug)]
|
#[derive(Copy, Clone, PartialEq, Eq, Debug)]
|
||||||
pub enum CompressionMethod {
|
pub enum CompressionMethod {
|
||||||
/// Store the file as is
|
/// Store the file as is
|
||||||
|
|
121
src/types.rs
121
src/types.rs
|
@ -1,5 +1,8 @@
|
||||||
//! Types that specify what is contained in a ZIP.
|
//! Types that specify what is contained in a ZIP.
|
||||||
|
|
||||||
|
#[cfg(feature = "time")]
|
||||||
|
use time::{error::ComponentRange, Date, Month, OffsetDateTime, PrimitiveDateTime, Time};
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug, PartialEq)]
|
#[derive(Clone, Copy, Debug, PartialEq)]
|
||||||
pub enum System {
|
pub enum System {
|
||||||
Dos = 0,
|
Dos = 0,
|
||||||
|
@ -117,30 +120,18 @@ impl DateTime {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "time")]
|
#[cfg(feature = "time")]
|
||||||
/// Converts a ::time::Tm object to a DateTime
|
/// Converts a OffsetDateTime object to a DateTime
|
||||||
///
|
///
|
||||||
/// Returns `Err` when this object is out of bounds
|
/// Returns `Err` when this object is out of bounds
|
||||||
pub fn from_time(tm: ::time::Tm) -> Result<DateTime, ()> {
|
pub fn from_time(dt: OffsetDateTime) -> Result<DateTime, ()> {
|
||||||
if tm.tm_year >= 80
|
if dt.year() >= 1980 && dt.year() <= 2107 {
|
||||||
&& tm.tm_year <= 207
|
|
||||||
&& tm.tm_mon >= 0
|
|
||||||
&& tm.tm_mon <= 11
|
|
||||||
&& tm.tm_mday >= 1
|
|
||||||
&& tm.tm_mday <= 31
|
|
||||||
&& tm.tm_hour >= 0
|
|
||||||
&& tm.tm_hour <= 23
|
|
||||||
&& tm.tm_min >= 0
|
|
||||||
&& tm.tm_min <= 59
|
|
||||||
&& tm.tm_sec >= 0
|
|
||||||
&& tm.tm_sec <= 60
|
|
||||||
{
|
|
||||||
Ok(DateTime {
|
Ok(DateTime {
|
||||||
year: (tm.tm_year + 1900) as u16,
|
year: (dt.year()) as u16,
|
||||||
month: (tm.tm_mon + 1) as u8,
|
month: (dt.month()) as u8,
|
||||||
day: tm.tm_mday as u8,
|
day: dt.day() as u8,
|
||||||
hour: tm.tm_hour as u8,
|
hour: dt.hour() as u8,
|
||||||
minute: tm.tm_min as u8,
|
minute: dt.minute() as u8,
|
||||||
second: tm.tm_sec as u8,
|
second: dt.second() as u8,
|
||||||
})
|
})
|
||||||
} else {
|
} else {
|
||||||
Err(())
|
Err(())
|
||||||
|
@ -158,20 +149,14 @@ impl DateTime {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "time")]
|
#[cfg(feature = "time")]
|
||||||
/// Converts the datetime to a Tm structure
|
/// Converts the DateTime to a OffsetDateTime structure
|
||||||
///
|
pub fn to_time(&self) -> Result<OffsetDateTime, ComponentRange> {
|
||||||
/// The fields `tm_wday`, `tm_yday`, `tm_utcoff` and `tm_nsec` are set to their defaults.
|
use std::convert::TryFrom;
|
||||||
pub fn to_time(&self) -> ::time::Tm {
|
|
||||||
::time::Tm {
|
let date =
|
||||||
tm_sec: self.second as i32,
|
Date::from_calendar_date(self.year as i32, Month::try_from(self.month)?, self.day)?;
|
||||||
tm_min: self.minute as i32,
|
let time = Time::from_hms(self.hour, self.minute, self.second)?;
|
||||||
tm_hour: self.hour as i32,
|
Ok(PrimitiveDateTime::new(date, time).assume_utc())
|
||||||
tm_mday: self.day as i32,
|
|
||||||
tm_mon: self.month as i32 - 1,
|
|
||||||
tm_year: self.year as i32 - 1900,
|
|
||||||
tm_isdst: -1,
|
|
||||||
..::time::empty_tm()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get the year. There is no epoch, i.e. 2018 will be returned as 2018.
|
/// Get the year. There is no epoch, i.e. 2018 will be returned as 2018.
|
||||||
|
@ -374,58 +359,26 @@ 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());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(feature = "time")]
|
||||||
|
use time::{format_description::well_known::Rfc3339, OffsetDateTime};
|
||||||
|
|
||||||
#[cfg(feature = "time")]
|
#[cfg(feature = "time")]
|
||||||
#[test]
|
#[test]
|
||||||
fn datetime_from_time_bounds() {
|
fn datetime_from_time_bounds() {
|
||||||
use super::DateTime;
|
use super::DateTime;
|
||||||
|
use time::macros::datetime;
|
||||||
|
|
||||||
// 1979-12-31 23:59:59
|
// 1979-12-31 23:59:59
|
||||||
assert!(DateTime::from_time(::time::Tm {
|
assert!(DateTime::from_time(datetime!(1979-12-31 23:59:59 UTC)).is_err());
|
||||||
tm_sec: 59,
|
|
||||||
tm_min: 59,
|
|
||||||
tm_hour: 23,
|
|
||||||
tm_mday: 31,
|
|
||||||
tm_mon: 11, // tm_mon has number range [0, 11]
|
|
||||||
tm_year: 79, // 1979 - 1900 = 79
|
|
||||||
..::time::empty_tm()
|
|
||||||
})
|
|
||||||
.is_err());
|
|
||||||
|
|
||||||
// 1980-01-01 00:00:00
|
// 1980-01-01 00:00:00
|
||||||
assert!(DateTime::from_time(::time::Tm {
|
assert!(DateTime::from_time(datetime!(1980-01-01 00:00:00 UTC)).is_ok());
|
||||||
tm_sec: 0,
|
|
||||||
tm_min: 0,
|
|
||||||
tm_hour: 0,
|
|
||||||
tm_mday: 1,
|
|
||||||
tm_mon: 0, // tm_mon has number range [0, 11]
|
|
||||||
tm_year: 80, // 1980 - 1900 = 80
|
|
||||||
..::time::empty_tm()
|
|
||||||
})
|
|
||||||
.is_ok());
|
|
||||||
|
|
||||||
// 2107-12-31 23:59:59
|
// 2107-12-31 23:59:59
|
||||||
assert!(DateTime::from_time(::time::Tm {
|
assert!(DateTime::from_time(datetime!(2107-12-31 23:59:59 UTC)).is_ok());
|
||||||
tm_sec: 59,
|
|
||||||
tm_min: 59,
|
|
||||||
tm_hour: 23,
|
|
||||||
tm_mday: 31,
|
|
||||||
tm_mon: 11, // tm_mon has number range [0, 11]
|
|
||||||
tm_year: 207, // 2107 - 1900 = 207
|
|
||||||
..::time::empty_tm()
|
|
||||||
})
|
|
||||||
.is_ok());
|
|
||||||
|
|
||||||
// 2108-01-01 00:00:00
|
// 2108-01-01 00:00:00
|
||||||
assert!(DateTime::from_time(::time::Tm {
|
assert!(DateTime::from_time(datetime!(2108-01-01 00:00:00 UTC)).is_err());
|
||||||
tm_sec: 0,
|
|
||||||
tm_min: 0,
|
|
||||||
tm_hour: 0,
|
|
||||||
tm_mday: 1,
|
|
||||||
tm_mon: 0, // tm_mon has number range [0, 11]
|
|
||||||
tm_year: 208, // 2108 - 1900 = 208
|
|
||||||
..::time::empty_tm()
|
|
||||||
})
|
|
||||||
.is_err());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -441,7 +394,7 @@ mod test {
|
||||||
|
|
||||||
#[cfg(feature = "time")]
|
#[cfg(feature = "time")]
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
format!("{}", dt.to_time().rfc3339()),
|
format!("{}", dt.to_time().unwrap().format(&Rfc3339).unwrap()),
|
||||||
"2018-11-17T10:38:30Z"
|
"2018-11-17T10:38:30Z"
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -458,10 +411,7 @@ mod test {
|
||||||
assert_eq!(dt.second(), 62);
|
assert_eq!(dt.second(), 62);
|
||||||
|
|
||||||
#[cfg(feature = "time")]
|
#[cfg(feature = "time")]
|
||||||
assert_eq!(
|
assert!(dt.to_time().is_err());
|
||||||
format!("{}", dt.to_time().rfc3339()),
|
|
||||||
"2107-15-31T31:63:62Z"
|
|
||||||
);
|
|
||||||
|
|
||||||
let dt = DateTime::from_msdos(0x0000, 0x0000);
|
let dt = DateTime::from_msdos(0x0000, 0x0000);
|
||||||
assert_eq!(dt.year(), 1980);
|
assert_eq!(dt.year(), 1980);
|
||||||
|
@ -472,10 +422,7 @@ mod test {
|
||||||
assert_eq!(dt.second(), 0);
|
assert_eq!(dt.second(), 0);
|
||||||
|
|
||||||
#[cfg(feature = "time")]
|
#[cfg(feature = "time")]
|
||||||
assert_eq!(
|
assert!(dt.to_time().is_err());
|
||||||
format!("{}", dt.to_time().rfc3339()),
|
|
||||||
"1980-00-00T00:00:00Z"
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "time")]
|
#[cfg(feature = "time")]
|
||||||
|
@ -484,8 +431,8 @@ mod test {
|
||||||
use super::DateTime;
|
use super::DateTime;
|
||||||
|
|
||||||
// 2020-01-01 00:00:00
|
// 2020-01-01 00:00:00
|
||||||
let clock = ::time::Timespec::new(1577836800, 0);
|
let clock = OffsetDateTime::from_unix_timestamp(1_577_836_800).unwrap();
|
||||||
let tm = ::time::at_utc(clock);
|
|
||||||
assert!(DateTime::from_time(tm).is_ok());
|
assert!(DateTime::from_time(clock).is_ok());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,6 +22,9 @@ use flate2::write::DeflateEncoder;
|
||||||
#[cfg(feature = "bzip2")]
|
#[cfg(feature = "bzip2")]
|
||||||
use bzip2::write::BzEncoder;
|
use bzip2::write::BzEncoder;
|
||||||
|
|
||||||
|
#[cfg(feature = "time")]
|
||||||
|
use time::OffsetDateTime;
|
||||||
|
|
||||||
enum GenericZipWriter<W: Write + io::Seek> {
|
enum GenericZipWriter<W: Write + io::Seek> {
|
||||||
Closed,
|
Closed,
|
||||||
Storer(W),
|
Storer(W),
|
||||||
|
@ -113,7 +116,7 @@ impl FileOptions {
|
||||||
)))]
|
)))]
|
||||||
compression_method: CompressionMethod::Stored,
|
compression_method: CompressionMethod::Stored,
|
||||||
#[cfg(feature = "time")]
|
#[cfg(feature = "time")]
|
||||||
last_modified_time: DateTime::from_time(time::now()).unwrap_or_default(),
|
last_modified_time: DateTime::from_time(OffsetDateTime::now_utc()).unwrap_or_default(),
|
||||||
#[cfg(not(feature = "time"))]
|
#[cfg(not(feature = "time"))]
|
||||||
last_modified_time: DateTime::default(),
|
last_modified_time: DateTime::default(),
|
||||||
permissions: None,
|
permissions: None,
|
||||||
|
|
Loading…
Add table
Reference in a new issue