Skip to content

Commit fe66485

Browse files
committed
add padding width
1 parent fa957cc commit fe66485

File tree

4 files changed

+333
-92
lines changed

4 files changed

+333
-92
lines changed

src/format/formatting.rs

+47-75
Original file line numberDiff line numberDiff line change
@@ -127,89 +127,61 @@ 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-
}
130+
// unpack padding width if provided
131+
let (spec, mut pad_width) = match spec {
132+
Numeric::Padded { numeric, width } => (numeric.as_ref().clone(), *width),
133+
numeric => (numeric.clone(), 0),
134+
};
133135

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)?,
136+
let (value, default_pad_width, is_year) = match (spec, self.date, self.time) {
137+
(Year, Some(d), _) => (d.year() as i64, 4, true),
138+
(YearDiv100, Some(d), _) => (d.year().div_euclid(100) as i64, 2, false),
139+
(YearMod100, Some(d), _) => (d.year().rem_euclid(100) as i64, 2, false),
140+
(IsoYear, Some(d), _) => (d.iso_week().year() as i64, 4, true),
141+
(IsoYearDiv100, Some(d), _) => (d.iso_week().year().div_euclid(100) as i64, 2, false),
142+
(IsoYearMod100, Some(d), _) => (d.iso_week().year().rem_euclid(100) as i64, 2, false),
143+
(Quarter, Some(d), _) => (d.quarter() as i64, 1, false),
144+
(Month, Some(d), _) => (d.month() as i64, 2, false),
145+
(Day, Some(d), _) => (d.day() as i64, 2, false),
146+
(WeekFromSun, Some(d), _) => (d.weeks_from(Weekday::Sun) as i64, 2, false),
147+
(WeekFromMon, Some(d), _) => (d.weeks_from(Weekday::Mon) as i64, 2, false),
148+
(IsoWeek, Some(d), _) => (d.iso_week().week() as i64, 2, false),
149+
(NumDaysFromSun, Some(d), _) => (d.weekday().num_days_from_sunday() as i64, 1, false),
150+
(WeekdayFromMon, Some(d), _) => (d.weekday().number_from_monday() as i64, 1, false),
151+
(Ordinal, Some(d), _) => (d.ordinal() as i64, 3, false),
152+
(Hour, _, Some(t)) => (t.hour() as i64, 2, false),
153+
(Hour12, _, Some(t)) => (t.hour12().1 as i64, 2, false),
154+
(Minute, _, Some(t)) => (t.minute() as i64, 2, false),
155+
(Second, _, Some(t)) => {
156+
((t.second() + t.nanosecond() / 1_000_000_000) as i64, 2, false)
140157
}
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))
158+
(Nanosecond, _, Some(t)) => ((t.nanosecond() % 1_000_000_000) as i64, 9, false),
159+
(Timestamp, Some(d), Some(t)) => {
160+
let offset = self.off.as_ref().map(|(_, o)| i64::from(o.local_minus_utc()));
161+
(d.and_time(t).and_utc().timestamp() - offset.unwrap_or(0), 9, false)
152162
}
153-
}
163+
(Internal(_), _, _) => return Ok(()), // for future expansion
164+
(Padded { .. }, _, _) => return Err(fmt::Error), // should be unwrapped above
165+
_ => return Err(fmt::Error), // insufficient arguments for given format
166+
};
154167

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-
}
168+
if pad_width == 0 {
169+
pad_width = default_pad_width;
175170
}
176171

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)
172+
let always_sign = is_year && !(0..10_000).contains(&value);
173+
if always_sign {
174+
match pad {
175+
Pad::None => write!(w, "{:+}", value),
176+
Pad::Zero => write!(w, "{:+01$}", value, pad_width + 1),
177+
Pad::Space => write!(w, "{:+1$}", value, pad_width + 1),
205178
}
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)
179+
} else {
180+
match pad {
181+
Pad::None => write!(w, "{}", value),
182+
Pad::Zero => write!(w, "{:01$}", value, pad_width),
183+
Pad::Space => write!(w, "{:1$}", value, pad_width),
210184
}
211-
(Internal(_), _, _) => Ok(()), // for future expansion
212-
_ => Err(fmt::Error), // insufficient arguments for given format
213185
}
214186
}
215187

src/format/mod.rs

+78
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,46 @@ 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+
impl Numeric {
171+
#[cfg(feature = "alloc")]
172+
fn with_padding(self, width: usize) -> Self {
173+
if width != 0 {
174+
// update padding
175+
match self {
176+
Numeric::Padded { numeric, .. } => Numeric::Padded { numeric, width },
177+
numeric => Numeric::Padded { numeric: Arc::new(numeric), width },
178+
}
179+
} else {
180+
// remove padding
181+
match self {
182+
Numeric::Padded { numeric, .. } => numeric.as_ref().clone(),
183+
numeric => numeric,
184+
}
185+
}
186+
}
187+
188+
#[cfg(not(feature = "alloc"))]
189+
fn with_padding(self, _width: usize) -> Self {
190+
self
191+
}
192+
}
193+
159194
/// An opaque type representing numeric item types for internal uses only.
160195
#[derive(Clone, Eq, Hash, PartialEq)]
161196
pub struct InternalNumeric {
@@ -555,3 +590,46 @@ impl FromStr for Month {
555590
}
556591
}
557592
}
593+
594+
#[cfg(test)]
595+
mod tests {
596+
use crate::format::*;
597+
598+
#[test]
599+
#[cfg(feature = "alloc")]
600+
fn test_numeric_with_padding() {
601+
// No padding
602+
assert_eq!(Numeric::Year.with_padding(0), Numeric::Year);
603+
604+
// Add padding
605+
assert_eq!(
606+
Numeric::Year.with_padding(5),
607+
Numeric::Padded { numeric: Arc::new(Numeric::Year), width: 5 }
608+
);
609+
610+
// Update padding
611+
assert_eq!(
612+
Numeric::Year.with_padding(5).with_padding(10),
613+
Numeric::Padded { numeric: Arc::new(Numeric::Year), width: 10 }
614+
);
615+
616+
// Remove padding
617+
assert_eq!(Numeric::Year.with_padding(5).with_padding(0), Numeric::Year);
618+
}
619+
620+
#[test]
621+
#[cfg(not(feature = "alloc"))]
622+
fn test_numeric_with_padding_disabled() {
623+
// No padding
624+
assert_eq!(Numeric::Year.with_padding(0), Numeric::Year);
625+
626+
// Add padding
627+
assert_eq!(Numeric::Year.with_padding(5), Numeric::Year);
628+
629+
// Update padding
630+
assert_eq!(Numeric::Year.with_padding(5).with_padding(10), Numeric::Year);
631+
632+
// Remove padding
633+
assert_eq!(Numeric::Year.with_padding(5).with_padding(0), Numeric::Year);
634+
}
635+
}

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)