1use alloc::string::String;
4use alloc::vec::Vec;
5use core::ops::Deref;
6use std::io;
7
8use num_conv::prelude::*;
9
10use crate::format_description::well_known::iso8601::EncodedConfig;
11use crate::format_description::well_known::{Iso8601, Rfc2822, Rfc3339};
12use crate::format_description::{BorrowedFormatItem, OwnedFormatItem};
13use crate::formatting::{
14 format_component, format_number_pad_zero, iso8601, write, MONTH_NAMES, WEEKDAY_NAMES,
15};
16use crate::{error, Date, Time, UtcOffset};
17
18#[cfg_attr(docsrs, doc(notable_trait))]
25pub trait Formattable: sealed::Sealed {}
26impl Formattable for BorrowedFormatItem<'_> {}
27impl Formattable for [BorrowedFormatItem<'_>] {}
28impl Formattable for OwnedFormatItem {}
29impl Formattable for [OwnedFormatItem] {}
30impl Formattable for Rfc3339 {}
31impl Formattable for Rfc2822 {}
32impl<const CONFIG: EncodedConfig> Formattable for Iso8601<CONFIG> {}
33impl<T: Deref> Formattable for T where T::Target: Formattable {}
34
35mod sealed {
37 #[allow(clippy::wildcard_imports)]
38 use super::*;
39
40 pub trait Sealed {
42 fn format_into(
44 &self,
45 output: &mut (impl io::Write + ?Sized),
46 date: Option<Date>,
47 time: Option<Time>,
48 offset: Option<UtcOffset>,
49 ) -> Result<usize, error::Format>;
50
51 fn format(
53 &self,
54 date: Option<Date>,
55 time: Option<Time>,
56 offset: Option<UtcOffset>,
57 ) -> Result<String, error::Format> {
58 let mut buf = Vec::new();
59 self.format_into(&mut buf, date, time, offset)?;
60 Ok(String::from_utf8_lossy(&buf).into_owned())
61 }
62 }
63}
64
65impl sealed::Sealed for BorrowedFormatItem<'_> {
66 fn format_into(
67 &self,
68 output: &mut (impl io::Write + ?Sized),
69 date: Option<Date>,
70 time: Option<Time>,
71 offset: Option<UtcOffset>,
72 ) -> Result<usize, error::Format> {
73 Ok(match *self {
74 Self::Literal(literal) => write(output, literal)?,
75 Self::Component(component) => format_component(output, component, date, time, offset)?,
76 Self::Compound(items) => items.format_into(output, date, time, offset)?,
77 Self::Optional(item) => item.format_into(output, date, time, offset)?,
78 Self::First(items) => match items {
79 [] => 0,
80 [item, ..] => item.format_into(output, date, time, offset)?,
81 },
82 })
83 }
84}
85
86impl sealed::Sealed for [BorrowedFormatItem<'_>] {
87 fn format_into(
88 &self,
89 output: &mut (impl io::Write + ?Sized),
90 date: Option<Date>,
91 time: Option<Time>,
92 offset: Option<UtcOffset>,
93 ) -> Result<usize, error::Format> {
94 let mut bytes = 0;
95 for item in self.iter() {
96 bytes += item.format_into(output, date, time, offset)?;
97 }
98 Ok(bytes)
99 }
100}
101
102impl sealed::Sealed for OwnedFormatItem {
103 fn format_into(
104 &self,
105 output: &mut (impl io::Write + ?Sized),
106 date: Option<Date>,
107 time: Option<Time>,
108 offset: Option<UtcOffset>,
109 ) -> Result<usize, error::Format> {
110 match self {
111 Self::Literal(literal) => Ok(write(output, literal)?),
112 Self::Component(component) => format_component(output, *component, date, time, offset),
113 Self::Compound(items) => items.format_into(output, date, time, offset),
114 Self::Optional(item) => item.format_into(output, date, time, offset),
115 Self::First(items) => match &**items {
116 [] => Ok(0),
117 [item, ..] => item.format_into(output, date, time, offset),
118 },
119 }
120 }
121}
122
123impl sealed::Sealed for [OwnedFormatItem] {
124 fn format_into(
125 &self,
126 output: &mut (impl io::Write + ?Sized),
127 date: Option<Date>,
128 time: Option<Time>,
129 offset: Option<UtcOffset>,
130 ) -> Result<usize, error::Format> {
131 let mut bytes = 0;
132 for item in self.iter() {
133 bytes += item.format_into(output, date, time, offset)?;
134 }
135 Ok(bytes)
136 }
137}
138
139impl<T: Deref> sealed::Sealed for T
140where
141 T::Target: sealed::Sealed,
142{
143 fn format_into(
144 &self,
145 output: &mut (impl io::Write + ?Sized),
146 date: Option<Date>,
147 time: Option<Time>,
148 offset: Option<UtcOffset>,
149 ) -> Result<usize, error::Format> {
150 self.deref().format_into(output, date, time, offset)
151 }
152}
153
154impl sealed::Sealed for Rfc2822 {
155 fn format_into(
156 &self,
157 output: &mut (impl io::Write + ?Sized),
158 date: Option<Date>,
159 time: Option<Time>,
160 offset: Option<UtcOffset>,
161 ) -> Result<usize, error::Format> {
162 let date = date.ok_or(error::Format::InsufficientTypeInformation)?;
163 let time = time.ok_or(error::Format::InsufficientTypeInformation)?;
164 let offset = offset.ok_or(error::Format::InsufficientTypeInformation)?;
165
166 let mut bytes = 0;
167
168 let (year, month, day) = date.to_calendar_date();
169
170 if year < 1900 {
171 return Err(error::Format::InvalidComponent("year"));
172 }
173 if offset.seconds_past_minute() != 0 {
174 return Err(error::Format::InvalidComponent("offset_second"));
175 }
176
177 bytes += write(
178 output,
179 &WEEKDAY_NAMES[date.weekday().number_days_from_monday().extend::<usize>()][..3],
180 )?;
181 bytes += write(output, b", ")?;
182 bytes += format_number_pad_zero::<2>(output, day)?;
183 bytes += write(output, b" ")?;
184 bytes += write(
185 output,
186 &MONTH_NAMES[u8::from(month).extend::<usize>() - 1][..3],
187 )?;
188 bytes += write(output, b" ")?;
189 bytes += format_number_pad_zero::<4>(output, year.cast_unsigned())?;
190 bytes += write(output, b" ")?;
191 bytes += format_number_pad_zero::<2>(output, time.hour())?;
192 bytes += write(output, b":")?;
193 bytes += format_number_pad_zero::<2>(output, time.minute())?;
194 bytes += write(output, b":")?;
195 bytes += format_number_pad_zero::<2>(output, time.second())?;
196 bytes += write(output, b" ")?;
197 bytes += write(output, if offset.is_negative() { b"-" } else { b"+" })?;
198 bytes += format_number_pad_zero::<2>(output, offset.whole_hours().unsigned_abs())?;
199 bytes += format_number_pad_zero::<2>(output, offset.minutes_past_hour().unsigned_abs())?;
200
201 Ok(bytes)
202 }
203}
204
205impl sealed::Sealed for Rfc3339 {
206 fn format_into(
207 &self,
208 output: &mut (impl io::Write + ?Sized),
209 date: Option<Date>,
210 time: Option<Time>,
211 offset: Option<UtcOffset>,
212 ) -> Result<usize, error::Format> {
213 let date = date.ok_or(error::Format::InsufficientTypeInformation)?;
214 let time = time.ok_or(error::Format::InsufficientTypeInformation)?;
215 let offset = offset.ok_or(error::Format::InsufficientTypeInformation)?;
216
217 let mut bytes = 0;
218
219 let year = date.year();
220
221 if !(0..10_000).contains(&year) {
222 return Err(error::Format::InvalidComponent("year"));
223 }
224 if offset.whole_hours().unsigned_abs() > 23 {
225 return Err(error::Format::InvalidComponent("offset_hour"));
226 }
227 if offset.seconds_past_minute() != 0 {
228 return Err(error::Format::InvalidComponent("offset_second"));
229 }
230
231 bytes += format_number_pad_zero::<4>(output, year.cast_unsigned())?;
232 bytes += write(output, b"-")?;
233 bytes += format_number_pad_zero::<2>(output, u8::from(date.month()))?;
234 bytes += write(output, b"-")?;
235 bytes += format_number_pad_zero::<2>(output, date.day())?;
236 bytes += write(output, b"T")?;
237 bytes += format_number_pad_zero::<2>(output, time.hour())?;
238 bytes += write(output, b":")?;
239 bytes += format_number_pad_zero::<2>(output, time.minute())?;
240 bytes += write(output, b":")?;
241 bytes += format_number_pad_zero::<2>(output, time.second())?;
242
243 if time.nanosecond() != 0 {
244 let nanos = time.nanosecond();
245 bytes += write(output, b".")?;
246 bytes += if nanos % 10 != 0 {
247 format_number_pad_zero::<9>(output, nanos)
248 } else if (nanos / 10) % 10 != 0 {
249 format_number_pad_zero::<8>(output, nanos / 10)
250 } else if (nanos / 100) % 10 != 0 {
251 format_number_pad_zero::<7>(output, nanos / 100)
252 } else if (nanos / 1_000) % 10 != 0 {
253 format_number_pad_zero::<6>(output, nanos / 1_000)
254 } else if (nanos / 10_000) % 10 != 0 {
255 format_number_pad_zero::<5>(output, nanos / 10_000)
256 } else if (nanos / 100_000) % 10 != 0 {
257 format_number_pad_zero::<4>(output, nanos / 100_000)
258 } else if (nanos / 1_000_000) % 10 != 0 {
259 format_number_pad_zero::<3>(output, nanos / 1_000_000)
260 } else if (nanos / 10_000_000) % 10 != 0 {
261 format_number_pad_zero::<2>(output, nanos / 10_000_000)
262 } else {
263 format_number_pad_zero::<1>(output, nanos / 100_000_000)
264 }?;
265 }
266
267 if offset == UtcOffset::UTC {
268 bytes += write(output, b"Z")?;
269 return Ok(bytes);
270 }
271
272 bytes += write(output, if offset.is_negative() { b"-" } else { b"+" })?;
273 bytes += format_number_pad_zero::<2>(output, offset.whole_hours().unsigned_abs())?;
274 bytes += write(output, b":")?;
275 bytes += format_number_pad_zero::<2>(output, offset.minutes_past_hour().unsigned_abs())?;
276
277 Ok(bytes)
278 }
279}
280
281impl<const CONFIG: EncodedConfig> sealed::Sealed for Iso8601<CONFIG> {
282 fn format_into(
283 &self,
284 output: &mut (impl io::Write + ?Sized),
285 date: Option<Date>,
286 time: Option<Time>,
287 offset: Option<UtcOffset>,
288 ) -> Result<usize, error::Format> {
289 let mut bytes = 0;
290
291 if Self::FORMAT_DATE {
292 let date = date.ok_or(error::Format::InsufficientTypeInformation)?;
293 bytes += iso8601::format_date::<CONFIG>(output, date)?;
294 }
295 if Self::FORMAT_TIME {
296 let time = time.ok_or(error::Format::InsufficientTypeInformation)?;
297 bytes += iso8601::format_time::<CONFIG>(output, time)?;
298 }
299 if Self::FORMAT_OFFSET {
300 let offset = offset.ok_or(error::Format::InsufficientTypeInformation)?;
301 bytes += iso8601::format_offset::<CONFIG>(output, offset)?;
302 }
303
304 if bytes == 0 {
305 panic!("attempted to format a parsing-only format description");
308 }
309
310 Ok(bytes)
311 }
312}