refactor!: Rename from_msdos
to from_msdos_unchecked
, make it unsafe, and add try_from_msdos
(#145)
This commit is contained in:
parent
ef74a26dd5
commit
7a34aa5f41
3 changed files with 47 additions and 13 deletions
|
@ -1009,7 +1009,7 @@ fn central_header_to_zip_file_inner<R: Read>(
|
||||||
CompressionMethod::from_u16(compression_method)
|
CompressionMethod::from_u16(compression_method)
|
||||||
},
|
},
|
||||||
compression_level: None,
|
compression_level: None,
|
||||||
last_modified_time: DateTime::from_msdos(last_mod_date, last_mod_time),
|
last_modified_time: DateTime::try_from_msdos(last_mod_date, last_mod_time)?,
|
||||||
crc32,
|
crc32,
|
||||||
compressed_size: compressed_size as u64,
|
compressed_size: compressed_size as u64,
|
||||||
uncompressed_size: uncompressed_size as u64,
|
uncompressed_size: uncompressed_size as u64,
|
||||||
|
@ -1396,7 +1396,7 @@ pub fn read_zipfile_from_stream<'a, R: Read>(reader: &'a mut R) -> ZipResult<Opt
|
||||||
using_data_descriptor: false,
|
using_data_descriptor: false,
|
||||||
compression_method,
|
compression_method,
|
||||||
compression_level: None,
|
compression_level: None,
|
||||||
last_modified_time: DateTime::from_msdos(last_mod_date, last_mod_time),
|
last_modified_time: DateTime::try_from_msdos(last_mod_date, last_mod_time)?,
|
||||||
crc32,
|
crc32,
|
||||||
compressed_size: compressed_size as u64,
|
compressed_size: compressed_size as u64,
|
||||||
uncompressed_size: uncompressed_size as u64,
|
uncompressed_size: uncompressed_size as u64,
|
||||||
|
|
|
@ -62,6 +62,12 @@ impl From<ZipError> for io::Error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl From<DateTimeRangeError> for ZipError {
|
||||||
|
fn from(_: DateTimeRangeError) -> Self {
|
||||||
|
ZipError::InvalidArchive("Invalid date or time")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Error type for time parsing
|
/// Error type for time parsing
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct DateTimeRangeError;
|
pub struct DateTimeRangeError;
|
||||||
|
|
50
src/types.rs
50
src/types.rs
|
@ -63,11 +63,10 @@ impl From<System> for u8 {
|
||||||
/// # Warning
|
/// # Warning
|
||||||
///
|
///
|
||||||
/// Because there is no timezone associated with the [`DateTime`], they should ideally only
|
/// 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
|
/// be used for user-facing descriptions.
|
||||||
/// [`OffsetDateTime`] (which is the equivalent of chrono's `NaiveDateTime`).
|
|
||||||
///
|
///
|
||||||
/// Modern zip files store more precise timestamps, which are ignored by [`crate::read::ZipArchive`],
|
/// Modern zip files store more precise timestamps; see [`crate::extra_fields::ExtendedTimestamp`]
|
||||||
/// so keep in mind that these timestamps are unreliable. [We're working on this](https://github.com/zip-rs/zip/issues/156#issuecomment-652981904).
|
/// for details.
|
||||||
#[derive(Clone, Copy, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
|
#[derive(Clone, Copy, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
|
||||||
pub struct DateTime {
|
pub struct DateTime {
|
||||||
year: u16,
|
year: u16,
|
||||||
|
@ -149,7 +148,10 @@ impl fmt::Display for DateTime {
|
||||||
|
|
||||||
impl DateTime {
|
impl DateTime {
|
||||||
/// Converts an msdos (u16, u16) pair to a DateTime object
|
/// Converts an msdos (u16, u16) pair to a DateTime object
|
||||||
pub const fn from_msdos(datepart: u16, timepart: u16) -> DateTime {
|
///
|
||||||
|
/// # Safety
|
||||||
|
/// The caller must ensure the date and time are valid.
|
||||||
|
pub const unsafe fn from_msdos_unchecked(datepart: u16, timepart: u16) -> DateTime {
|
||||||
let seconds = (timepart & 0b0000000000011111) << 1;
|
let seconds = (timepart & 0b0000000000011111) << 1;
|
||||||
let minutes = (timepart & 0b0000011111100000) >> 5;
|
let minutes = (timepart & 0b0000011111100000) >> 5;
|
||||||
let hours = (timepart & 0b1111100000000000) >> 11;
|
let hours = (timepart & 0b1111100000000000) >> 11;
|
||||||
|
@ -167,6 +169,25 @@ impl DateTime {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Converts an msdos (u16, u16) pair to a DateTime object if it represents a valid date and
|
||||||
|
/// time.
|
||||||
|
pub fn try_from_msdos(datepart: u16, timepart: u16) -> Result<DateTime, DateTimeRangeError> {
|
||||||
|
let seconds = (timepart & 0b0000000000011111) << 1;
|
||||||
|
let minutes = (timepart & 0b0000011111100000) >> 5;
|
||||||
|
let hours = (timepart & 0b1111100000000000) >> 11;
|
||||||
|
let days = datepart & 0b0000000000011111;
|
||||||
|
let months = (datepart & 0b0000000111100000) >> 5;
|
||||||
|
let years = (datepart & 0b1111111000000000) >> 9;
|
||||||
|
Self::from_date_and_time(
|
||||||
|
years.checked_add(1980).ok_or(DateTimeRangeError)?,
|
||||||
|
months.try_into()?,
|
||||||
|
days.try_into()?,
|
||||||
|
hours.try_into()?,
|
||||||
|
minutes.try_into()?,
|
||||||
|
seconds.try_into()?,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
/// Constructs a DateTime from a specific date and time
|
/// Constructs a DateTime from a specific date and time
|
||||||
///
|
///
|
||||||
/// The bounds are:
|
/// The bounds are:
|
||||||
|
@ -748,7 +769,8 @@ mod test {
|
||||||
use super::DateTime;
|
use super::DateTime;
|
||||||
|
|
||||||
// 2018-11-17 10:38:30 UTC
|
// 2018-11-17 10:38:30 UTC
|
||||||
let dt = OffsetDateTime::try_from(DateTime::from_msdos(0x4D71, 0x54CF)).unwrap();
|
let dt =
|
||||||
|
OffsetDateTime::try_from(DateTime::try_from_msdos(0x4D71, 0x54CF).unwrap()).unwrap();
|
||||||
assert_eq!(dt, datetime!(2018-11-17 10:38:30 UTC));
|
assert_eq!(dt, datetime!(2018-11-17 10:38:30 UTC));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -758,17 +780,23 @@ mod test {
|
||||||
use super::DateTime;
|
use super::DateTime;
|
||||||
|
|
||||||
// 1980-00-00 00:00:00
|
// 1980-00-00 00:00:00
|
||||||
assert!(OffsetDateTime::try_from(DateTime::from_msdos(0x0000, 0x0000)).is_err());
|
assert!(OffsetDateTime::try_from(unsafe {
|
||||||
|
DateTime::from_msdos_unchecked(0x0000, 0x0000)
|
||||||
|
})
|
||||||
|
.is_err());
|
||||||
|
|
||||||
// 2107-15-31 31:63:62
|
// 2107-15-31 31:63:62
|
||||||
assert!(OffsetDateTime::try_from(DateTime::from_msdos(0xFFFF, 0xFFFF)).is_err());
|
assert!(OffsetDateTime::try_from(unsafe {
|
||||||
|
DateTime::from_msdos_unchecked(0xFFFF, 0xFFFF)
|
||||||
|
})
|
||||||
|
.is_err());
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
#[allow(deprecated)]
|
#[allow(deprecated)]
|
||||||
fn time_conversion() {
|
fn time_conversion() {
|
||||||
use super::DateTime;
|
use super::DateTime;
|
||||||
let dt = DateTime::from_msdos(0x4D71, 0x54CF);
|
let dt = DateTime::try_from_msdos(0x4D71, 0x54CF).unwrap();
|
||||||
assert_eq!(dt.year(), 2018);
|
assert_eq!(dt.year(), 2018);
|
||||||
assert_eq!(dt.month(), 11);
|
assert_eq!(dt.month(), 11);
|
||||||
assert_eq!(dt.day(), 17);
|
assert_eq!(dt.day(), 17);
|
||||||
|
@ -787,7 +815,7 @@ mod test {
|
||||||
#[allow(deprecated)]
|
#[allow(deprecated)]
|
||||||
fn time_out_of_bounds() {
|
fn time_out_of_bounds() {
|
||||||
use super::DateTime;
|
use super::DateTime;
|
||||||
let dt = DateTime::from_msdos(0xFFFF, 0xFFFF);
|
let dt = unsafe { DateTime::from_msdos_unchecked(0xFFFF, 0xFFFF) };
|
||||||
assert_eq!(dt.year(), 2107);
|
assert_eq!(dt.year(), 2107);
|
||||||
assert_eq!(dt.month(), 15);
|
assert_eq!(dt.month(), 15);
|
||||||
assert_eq!(dt.day(), 31);
|
assert_eq!(dt.day(), 31);
|
||||||
|
@ -798,7 +826,7 @@ mod test {
|
||||||
#[cfg(feature = "time")]
|
#[cfg(feature = "time")]
|
||||||
assert!(dt.to_time().is_err());
|
assert!(dt.to_time().is_err());
|
||||||
|
|
||||||
let dt = DateTime::from_msdos(0x0000, 0x0000);
|
let dt = unsafe { DateTime::from_msdos_unchecked(0x0000, 0x0000) };
|
||||||
assert_eq!(dt.year(), 1980);
|
assert_eq!(dt.year(), 1980);
|
||||||
assert_eq!(dt.month(), 0);
|
assert_eq!(dt.month(), 0);
|
||||||
assert_eq!(dt.day(), 0);
|
assert_eq!(dt.day(), 0);
|
||||||
|
|
Loading…
Add table
Reference in a new issue