Skip to content

Commit d89f8f9

Browse files
committed
add padding width
1 parent fa957cc commit d89f8f9

File tree

4 files changed

+356
-93
lines changed

4 files changed

+356
-93
lines changed

src/format/formatting.rs

+45-76
Original file line numberDiff line numberDiff line change
@@ -127,89 +127,58 @@ impl<'a, I: Iterator<Item = B> + Clone, B: Borrow<Item<'a>>> DelayedFormat<I> {
127127
fn format_numeric(&self, w: &mut impl Write, spec: &Numeric, pad: Pad) -> fmt::Result {
128128
use self::Numeric::*;
129129

130-
fn write_one(w: &mut impl Write, v: u8) -> fmt::Result {
131-
w.write_char((b'0' + v) as char)
132-
}
133-
134-
fn write_two(w: &mut impl Write, v: u8, pad: Pad) -> fmt::Result {
135-
let ones = b'0' + v % 10;
136-
match (v / 10, pad) {
137-
(0, Pad::None) => {}
138-
(0, Pad::Space) => w.write_char(' ')?,
139-
(tens, _) => w.write_char((b'0' + tens) as char)?,
130+
// unpack padding width if provided
131+
let (spec, mut pad_width) = spec.unwrap_padding();
132+
133+
let (value, default_pad_width, is_year) = match (spec, self.date, self.time) {
134+
(Year, Some(d), _) => (d.year() as i64, 4, true),
135+
(YearDiv100, Some(d), _) => (d.year().div_euclid(100) as i64, 2, false),
136+
(YearMod100, Some(d), _) => (d.year().rem_euclid(100) as i64, 2, false),
137+
(IsoYear, Some(d), _) => (d.iso_week().year() as i64, 4, true),
138+
(IsoYearDiv100, Some(d), _) => (d.iso_week().year().div_euclid(100) as i64, 2, false),
139+
(IsoYearMod100, Some(d), _) => (d.iso_week().year().rem_euclid(100) as i64, 2, false),
140+
(Quarter, Some(d), _) => (d.quarter() as i64, 1, false),
141+
(Month, Some(d), _) => (d.month() as i64, 2, false),
142+
(Day, Some(d), _) => (d.day() as i64, 2, false),
143+
(WeekFromSun, Some(d), _) => (d.weeks_from(Weekday::Sun) as i64, 2, false),
144+
(WeekFromMon, Some(d), _) => (d.weeks_from(Weekday::Mon) as i64, 2, false),
145+
(IsoWeek, Some(d), _) => (d.iso_week().week() as i64, 2, false),
146+
(NumDaysFromSun, Some(d), _) => (d.weekday().num_days_from_sunday() as i64, 1, false),
147+
(WeekdayFromMon, Some(d), _) => (d.weekday().number_from_monday() as i64, 1, false),
148+
(Ordinal, Some(d), _) => (d.ordinal() as i64, 3, false),
149+
(Hour, _, Some(t)) => (t.hour() as i64, 2, false),
150+
(Hour12, _, Some(t)) => (t.hour12().1 as i64, 2, false),
151+
(Minute, _, Some(t)) => (t.minute() as i64, 2, false),
152+
(Second, _, Some(t)) => {
153+
((t.second() + t.nanosecond() / 1_000_000_000) as i64, 2, false)
140154
}
141-
w.write_char(ones as char)
142-
}
143-
144-
#[inline]
145-
fn write_year(w: &mut impl Write, year: i32, pad: Pad) -> fmt::Result {
146-
if (1000..=9999).contains(&year) {
147-
// fast path
148-
write_hundreds(w, (year / 100) as u8)?;
149-
write_hundreds(w, (year % 100) as u8)
150-
} else {
151-
write_n(w, 4, year as i64, pad, !(0..10_000).contains(&year))
155+
(Nanosecond, _, Some(t)) => ((t.nanosecond() % 1_000_000_000) as i64, 9, false),
156+
(Timestamp, Some(d), Some(t)) => {
157+
let offset = self.off.as_ref().map(|(_, o)| i64::from(o.local_minus_utc()));
158+
(d.and_time(t).and_utc().timestamp() - offset.unwrap_or(0), 9, false)
152159
}
153-
}
160+
(Internal(_), _, _) => return Ok(()), // for future expansion
161+
(Padded { .. }, _, _) => return Err(fmt::Error), // should be unwrapped above
162+
_ => return Err(fmt::Error), // insufficient arguments for given format
163+
};
154164

155-
fn write_n(
156-
w: &mut impl Write,
157-
n: usize,
158-
v: i64,
159-
pad: Pad,
160-
always_sign: bool,
161-
) -> fmt::Result {
162-
if always_sign {
163-
match pad {
164-
Pad::None => write!(w, "{:+}", v),
165-
Pad::Zero => write!(w, "{:+01$}", v, n + 1),
166-
Pad::Space => write!(w, "{:+1$}", v, n + 1),
167-
}
168-
} else {
169-
match pad {
170-
Pad::None => write!(w, "{}", v),
171-
Pad::Zero => write!(w, "{:01$}", v, n),
172-
Pad::Space => write!(w, "{:1$}", v, n),
173-
}
174-
}
165+
if pad_width == 0 {
166+
pad_width = default_pad_width;
175167
}
176168

177-
match (spec, self.date, self.time) {
178-
(Year, Some(d), _) => write_year(w, d.year(), pad),
179-
(YearDiv100, Some(d), _) => write_two(w, d.year().div_euclid(100) as u8, pad),
180-
(YearMod100, Some(d), _) => write_two(w, d.year().rem_euclid(100) as u8, pad),
181-
(IsoYear, Some(d), _) => write_year(w, d.iso_week().year(), pad),
182-
(IsoYearDiv100, Some(d), _) => {
183-
write_two(w, d.iso_week().year().div_euclid(100) as u8, pad)
184-
}
185-
(IsoYearMod100, Some(d), _) => {
186-
write_two(w, d.iso_week().year().rem_euclid(100) as u8, pad)
187-
}
188-
(Quarter, Some(d), _) => write_one(w, d.quarter() as u8),
189-
(Month, Some(d), _) => write_two(w, d.month() as u8, pad),
190-
(Day, Some(d), _) => write_two(w, d.day() as u8, pad),
191-
(WeekFromSun, Some(d), _) => write_two(w, d.weeks_from(Weekday::Sun) as u8, pad),
192-
(WeekFromMon, Some(d), _) => write_two(w, d.weeks_from(Weekday::Mon) as u8, pad),
193-
(IsoWeek, Some(d), _) => write_two(w, d.iso_week().week() as u8, pad),
194-
(NumDaysFromSun, Some(d), _) => write_one(w, d.weekday().num_days_from_sunday() as u8),
195-
(WeekdayFromMon, Some(d), _) => write_one(w, d.weekday().number_from_monday() as u8),
196-
(Ordinal, Some(d), _) => write_n(w, 3, d.ordinal() as i64, pad, false),
197-
(Hour, _, Some(t)) => write_two(w, t.hour() as u8, pad),
198-
(Hour12, _, Some(t)) => write_two(w, t.hour12().1 as u8, pad),
199-
(Minute, _, Some(t)) => write_two(w, t.minute() as u8, pad),
200-
(Second, _, Some(t)) => {
201-
write_two(w, (t.second() + t.nanosecond() / 1_000_000_000) as u8, pad)
202-
}
203-
(Nanosecond, _, Some(t)) => {
204-
write_n(w, 9, (t.nanosecond() % 1_000_000_000) as i64, pad, false)
169+
let always_sign = is_year && !(0..10_000).contains(&value);
170+
if always_sign {
171+
match pad {
172+
Pad::None => write!(w, "{:+}", value),
173+
Pad::Zero => write!(w, "{:+01$}", value, pad_width + 1),
174+
Pad::Space => write!(w, "{:+1$}", value, pad_width + 1),
205175
}
206-
(Timestamp, Some(d), Some(t)) => {
207-
let offset = self.off.as_ref().map(|(_, o)| i64::from(o.local_minus_utc()));
208-
let timestamp = d.and_time(t).and_utc().timestamp() - offset.unwrap_or(0);
209-
write_n(w, 9, timestamp, pad, false)
176+
} else {
177+
match pad {
178+
Pad::None => write!(w, "{}", value),
179+
Pad::Zero => write!(w, "{:01$}", value, pad_width),
180+
Pad::Space => write!(w, "{:1$}", value, pad_width),
210181
}
211-
(Internal(_), _, _) => Ok(()), // for future expansion
212-
_ => Err(fmt::Error), // insufficient arguments for given format
213182
}
214183
}
215184

src/format/mod.rs

+103
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,8 @@
3333
3434
#[cfg(all(feature = "alloc", not(feature = "std"), not(test)))]
3535
use alloc::boxed::Box;
36+
#[cfg(feature = "alloc")]
37+
use alloc::sync::Arc;
3638
use core::fmt;
3739
use core::str::FromStr;
3840
#[cfg(feature = "std")]
@@ -149,13 +151,71 @@ pub enum Numeric {
149151
/// For formatting, it assumes UTC upon the absence of time zone offset.
150152
Timestamp,
151153

154+
#[cfg(feature = "alloc")]
155+
/// An extension to carry the width of the padding.
156+
Padded {
157+
/// The numeric to be padded.
158+
numeric: Arc<Numeric>,
159+
/// The width of the padding.
160+
width: usize,
161+
},
162+
152163
/// Internal uses only.
153164
///
154165
/// This item exists so that one can add additional internal-only formatting
155166
/// without breaking major compatibility (as enum variants cannot be selectively private).
156167
Internal(InternalNumeric),
157168
}
158169

170+
#[cfg(feature = "alloc")]
171+
impl Numeric {
172+
/// Adds the with of the padding to the numeric
173+
///
174+
/// Should be removed if the padding width is added to the `Pad` enum.
175+
pub fn with_padding(self, width: usize) -> Self {
176+
if width != 0 {
177+
// update padding
178+
match self {
179+
Numeric::Padded { numeric, .. } => Numeric::Padded { numeric, width },
180+
numeric => Numeric::Padded { numeric: Arc::new(numeric), width },
181+
}
182+
} else {
183+
// remove padding
184+
match self {
185+
Numeric::Padded { numeric, .. } => numeric.as_ref().clone(),
186+
numeric => numeric,
187+
}
188+
}
189+
}
190+
191+
/// Gets the numeric and padding width from the numeric
192+
///
193+
/// Should be removed if the padding width is added to the `Pad` enum.
194+
pub fn unwrap_padding(&self) -> (&Self, usize) {
195+
match self {
196+
Numeric::Padded { numeric, width } => (numeric.as_ref(), *width),
197+
numeric => (numeric, 0),
198+
}
199+
}
200+
}
201+
202+
#[cfg(not(feature = "alloc"))]
203+
impl Numeric {
204+
/// Adds the with of the padding to the numeric
205+
///
206+
/// Should be removed if the padding width is added to the `Pad` enum.
207+
pub fn with_padding(self, _width: usize) -> Self {
208+
self
209+
}
210+
211+
/// Gets the numeric and padding width from the numeric
212+
///
213+
/// Should be removed if the padding width is added to the `Pad` enum.
214+
pub fn unwrap_padding(&self) -> (&Self, usize) {
215+
(self, 0)
216+
}
217+
}
218+
159219
/// An opaque type representing numeric item types for internal uses only.
160220
#[derive(Clone, Eq, Hash, PartialEq)]
161221
pub struct InternalNumeric {
@@ -555,3 +615,46 @@ impl FromStr for Month {
555615
}
556616
}
557617
}
618+
619+
#[cfg(test)]
620+
mod tests {
621+
use crate::format::*;
622+
623+
#[test]
624+
#[cfg(feature = "alloc")]
625+
fn test_numeric_with_padding() {
626+
// No padding
627+
assert_eq!(Numeric::Year.with_padding(0), Numeric::Year);
628+
629+
// Add padding
630+
assert_eq!(
631+
Numeric::Year.with_padding(5),
632+
Numeric::Padded { numeric: Arc::new(Numeric::Year), width: 5 }
633+
);
634+
635+
// Update padding
636+
assert_eq!(
637+
Numeric::Year.with_padding(5).with_padding(10),
638+
Numeric::Padded { numeric: Arc::new(Numeric::Year), width: 10 }
639+
);
640+
641+
// Remove padding
642+
assert_eq!(Numeric::Year.with_padding(5).with_padding(0), Numeric::Year);
643+
}
644+
645+
#[test]
646+
#[cfg(not(feature = "alloc"))]
647+
fn test_numeric_with_padding_disabled() {
648+
// No padding
649+
assert_eq!(Numeric::Year.with_padding(0), Numeric::Year);
650+
651+
// Add padding
652+
assert_eq!(Numeric::Year.with_padding(5), Numeric::Year);
653+
654+
// Update padding
655+
assert_eq!(Numeric::Year.with_padding(5).with_padding(10), Numeric::Year);
656+
657+
// Remove padding
658+
assert_eq!(Numeric::Year.with_padding(5).with_padding(0), Numeric::Year);
659+
}
660+
}

src/format/parse.rs

+87-1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,8 @@
77
use core::borrow::Borrow;
88
use core::str;
99

10+
#[cfg(feature = "alloc")]
11+
use super::ParseErrorKind::BadFormat;
1012
use super::scan;
1113
use super::{BAD_FORMAT, INVALID, OUT_OF_RANGE, TOO_LONG, TOO_SHORT};
1214
use super::{Fixed, InternalFixed, InternalInternal, Item, Numeric, Pad, Parsed};
@@ -359,7 +361,9 @@ where
359361
Nanosecond => (9, false, Parsed::set_nanosecond),
360362
Timestamp => (usize::MAX, false, Parsed::set_timestamp),
361363

362-
// for the future expansion
364+
#[cfg(feature = "alloc")]
365+
Padded { .. } => return Err(ParseError(BadFormat)),
366+
363367
Internal(ref int) => match int._dummy {},
364368
};
365369

@@ -1580,6 +1584,88 @@ mod tests {
15801584
);
15811585
}
15821586

1587+
#[test]
1588+
#[rustfmt::skip]
1589+
#[cfg(feature = "alloc")]
1590+
fn test_parse_padded() {
1591+
use crate::format::InternalInternal::*;
1592+
use crate::format::Item::{Literal, Space};
1593+
use crate::format::Numeric::*;
1594+
use crate::format::ParseErrorKind::BadFormat;
1595+
1596+
check(
1597+
"2000-01-02 03:04:05Z",
1598+
&[
1599+
nums(Year.with_padding(5))
1600+
],
1601+
Err(ParseError(BadFormat)),
1602+
);
1603+
1604+
check(
1605+
"2000-01-02 03:04:05Z",
1606+
&[
1607+
nums(Year.with_padding(5)), Literal("-"), num(Month), Literal("-"), num(Day), Space(" "),
1608+
num(Hour), Literal(":"), num(Minute), Literal(":"), num(Second),
1609+
internal_fixed(TimezoneOffsetPermissive)
1610+
],
1611+
Err(ParseError(BadFormat)),
1612+
);
1613+
1614+
check(
1615+
"2000-01-02 03:04:05Z",
1616+
&[
1617+
num(Year), Literal("-"), num(Month), Literal("-"), nums(Day.with_padding(5)), Space(" "),
1618+
num(Hour), Literal(":"), num(Minute), Literal(":"), num(Second),
1619+
internal_fixed(TimezoneOffsetPermissive)
1620+
],
1621+
Err(ParseError(BadFormat)),
1622+
);
1623+
}
1624+
1625+
#[test]
1626+
#[rustfmt::skip]
1627+
#[cfg(not(feature = "alloc"))]
1628+
fn test_parse_padded_disabled() {
1629+
use crate::format::InternalInternal::*;
1630+
use crate::format::Item::{Literal, Space};
1631+
use crate::format::Numeric::*;
1632+
use crate::format::ParseErrorKind::TooLong;
1633+
1634+
check(
1635+
"2000-01-02 03:04:05Z",
1636+
&[
1637+
nums(Year.with_padding(5))
1638+
],
1639+
Err(ParseError(TooLong)),
1640+
);
1641+
1642+
check(
1643+
"2000-01-02 03:04:05Z",
1644+
&[
1645+
nums(Year.with_padding(5)), Literal("-"), num(Month), Literal("-"), num(Day), Space(" "),
1646+
num(Hour), Literal(":"), num(Minute), Literal(":"), num(Second),
1647+
internal_fixed(TimezoneOffsetPermissive)
1648+
],
1649+
parsed!(
1650+
year: 2000, month: 1, day: 2, hour_div_12: 0, hour_mod_12: 3, minute: 4, second: 5,
1651+
offset: 0
1652+
),
1653+
);
1654+
1655+
check(
1656+
"2000-01-02 03:04:05Z",
1657+
&[
1658+
num(Year), Literal("-"), num(Month), Literal("-"), nums(Day.with_padding(5)), Space(" "),
1659+
num(Hour), Literal(":"), num(Minute), Literal(":"), num(Second),
1660+
internal_fixed(TimezoneOffsetPermissive)
1661+
],
1662+
parsed!(
1663+
year: 2000, month: 1, day: 2, hour_div_12: 0, hour_mod_12: 3, minute: 4, second: 5,
1664+
offset: 0
1665+
),
1666+
);
1667+
}
1668+
15831669
#[track_caller]
15841670
fn parses(s: &str, items: &[Item]) {
15851671
let mut parsed = Parsed::new();

0 commit comments

Comments
 (0)