writeable/
impls.rs
1use crate::*;
6use core::fmt;
7
8macro_rules! impl_write_num {
9 ($u:ty, $i:ty, $test:ident $(,$random_call:ident)?) => {
11 impl $crate::Writeable for $u {
12 fn write_to<W: core::fmt::Write + ?Sized>(&self, sink: &mut W) -> core::fmt::Result {
13 const MAX_LEN: usize = <$u>::MAX.ilog10() as usize + 1;
14 let mut buf = [b'0'; MAX_LEN];
15 let mut n = *self;
16 let mut i = MAX_LEN;
17 #[allow(clippy::indexing_slicing)] while n != 0 {
19 i -= 1;
20 buf[i] = b'0' + (n % 10) as u8;
21 n /= 10;
22 }
23 if i == MAX_LEN {
24 debug_assert_eq!(*self, 0);
25 i -= 1;
26 }
27 #[allow(clippy::indexing_slicing)] let s = unsafe { core::str::from_utf8_unchecked(&buf[i..]) };
29 sink.write_str(s)
30 }
31
32 fn writeable_length_hint(&self) -> $crate::LengthHint {
33 LengthHint::exact(self.checked_ilog10().unwrap_or(0) as usize + 1)
34 }
35 }
36
37 impl $crate::Writeable for $i {
38 fn write_to<W: core::fmt::Write + ?Sized>(&self, sink: &mut W) -> core::fmt::Result {
39 if self.is_negative() {
40 sink.write_str("-")?;
41 }
42 self.unsigned_abs().write_to(sink)
43 }
44
45 fn writeable_length_hint(&self) -> $crate::LengthHint {
46 $crate::LengthHint::exact(if self.is_negative() { 1 } else { 0 })
47 + self.unsigned_abs().writeable_length_hint()
48 }
49 }
50
51 #[test]
52 fn $test() {
53 use $crate::assert_writeable_eq;
54 assert_writeable_eq!(&(0 as $u), "0");
55 assert_writeable_eq!(&(0 as $i), "0");
56 assert_writeable_eq!(&(-0 as $i), "0");
57 assert_writeable_eq!(&(1 as $u), "1");
58 assert_writeable_eq!(&(1 as $i), "1");
59 assert_writeable_eq!(&(-1 as $i), "-1");
60 assert_writeable_eq!(&(9 as $u), "9");
61 assert_writeable_eq!(&(9 as $i), "9");
62 assert_writeable_eq!(&(-9 as $i), "-9");
63 assert_writeable_eq!(&(10 as $u), "10");
64 assert_writeable_eq!(&(10 as $i), "10");
65 assert_writeable_eq!(&(-10 as $i), "-10");
66 assert_writeable_eq!(&(99 as $u), "99");
67 assert_writeable_eq!(&(99 as $i), "99");
68 assert_writeable_eq!(&(-99 as $i), "-99");
69 assert_writeable_eq!(&(100 as $u), "100");
70 assert_writeable_eq!(&(-100 as $i), "-100");
71 assert_writeable_eq!(&<$u>::MAX, <$u>::MAX.to_string());
72 assert_writeable_eq!(&<$i>::MAX, <$i>::MAX.to_string());
73 assert_writeable_eq!(&<$i>::MIN, <$i>::MIN.to_string());
74
75 $(
76
77 use rand::{rngs::SmallRng, Rng, SeedableRng};
78 let mut rng = SmallRng::seed_from_u64(4); for _ in 0..1000 {
81 let rand = rng.$random_call::<$u>();
82 assert_writeable_eq!(rand, rand.to_string());
83 }
84 )?
85 }
86 };
87}
88
89impl_write_num!(u8, i8, test_u8, random);
90impl_write_num!(u16, i16, test_u16, random);
91impl_write_num!(u32, i32, test_u32, random);
92impl_write_num!(u64, i64, test_u64, random);
93impl_write_num!(u128, i128, test_u128, random);
94impl_write_num!(usize, isize, test_usize);
95
96impl Writeable for str {
97 #[inline]
98 fn write_to<W: fmt::Write + ?Sized>(&self, sink: &mut W) -> fmt::Result {
99 sink.write_str(self)
100 }
101
102 #[inline]
103 fn writeable_length_hint(&self) -> LengthHint {
104 LengthHint::exact(self.len())
105 }
106
107 #[inline]
119 fn write_to_string(&self) -> Cow<str> {
120 Cow::Borrowed(self)
121 }
122}
123
124impl Writeable for String {
125 #[inline]
126 fn write_to<W: fmt::Write + ?Sized>(&self, sink: &mut W) -> fmt::Result {
127 sink.write_str(self)
128 }
129
130 #[inline]
131 fn writeable_length_hint(&self) -> LengthHint {
132 LengthHint::exact(self.len())
133 }
134
135 #[inline]
136 fn write_to_string(&self) -> Cow<str> {
137 Cow::Borrowed(self)
138 }
139}
140
141impl Writeable for char {
142 #[inline]
143 fn write_to<W: fmt::Write + ?Sized>(&self, sink: &mut W) -> fmt::Result {
144 sink.write_char(*self)
145 }
146
147 #[inline]
148 fn writeable_length_hint(&self) -> LengthHint {
149 LengthHint::exact(self.len_utf8())
150 }
151
152 #[inline]
153 fn write_to_string(&self) -> Cow<str> {
154 let mut s = String::with_capacity(self.len_utf8());
155 s.push(*self);
156 Cow::Owned(s)
157 }
158}
159
160impl<T: Writeable + ?Sized> Writeable for &T {
161 #[inline]
162 fn write_to<W: fmt::Write + ?Sized>(&self, sink: &mut W) -> fmt::Result {
163 (*self).write_to(sink)
164 }
165
166 #[inline]
167 fn write_to_parts<W: PartsWrite + ?Sized>(&self, sink: &mut W) -> fmt::Result {
168 (*self).write_to_parts(sink)
169 }
170
171 #[inline]
172 fn writeable_length_hint(&self) -> LengthHint {
173 (*self).writeable_length_hint()
174 }
175
176 #[inline]
177 fn write_to_string(&self) -> Cow<str> {
178 (*self).write_to_string()
179 }
180}
181
182macro_rules! impl_write_smart_pointer {
183 ($ty:path, T: $extra_bound:path) => {
184 impl<'a, T: ?Sized + Writeable + $extra_bound> Writeable for $ty {
185 #[inline]
186 fn write_to<W: fmt::Write + ?Sized>(&self, sink: &mut W) -> fmt::Result {
187 core::borrow::Borrow::<T>::borrow(self).write_to(sink)
188 }
189 #[inline]
190 fn write_to_parts<W: PartsWrite + ?Sized>(&self, sink: &mut W) -> fmt::Result {
191 core::borrow::Borrow::<T>::borrow(self).write_to_parts(sink)
192 }
193 #[inline]
194 fn writeable_length_hint(&self) -> LengthHint {
195 core::borrow::Borrow::<T>::borrow(self).writeable_length_hint()
196 }
197 #[inline]
198 fn write_to_string(&self) -> Cow<str> {
199 core::borrow::Borrow::<T>::borrow(self).write_to_string()
200 }
201 }
202 };
203 ($ty:path) => {
204 impl_write_smart_pointer!($ty, T: Writeable);
206 };
207}
208
209impl_write_smart_pointer!(Cow<'a, T>, T: alloc::borrow::ToOwned);
210impl_write_smart_pointer!(alloc::boxed::Box<T>);
211impl_write_smart_pointer!(alloc::rc::Rc<T>);
212impl_write_smart_pointer!(alloc::sync::Arc<T>);
213
214#[test]
215fn test_string_impls() {
216 fn check_writeable_slice<W: Writeable + core::fmt::Display>(writeables: &[W]) {
217 assert_writeable_eq!(&writeables[0], "");
218 assert_writeable_eq!(&writeables[1], "abc");
219 assert!(matches!(writeables[0].write_to_string(), Cow::Borrowed(_)));
220 assert!(matches!(writeables[1].write_to_string(), Cow::Borrowed(_)));
221 }
222
223 let arr: &[&str] = &["", "abc"];
225 check_writeable_slice(arr);
226
227 let arr: &[String] = &[String::new(), "abc".to_owned()];
229 check_writeable_slice(arr);
230
231 let chars = ['a', 'β', '你', '😀'];
233 for i in 0..chars.len() {
234 let s = String::from(chars[i]);
235 assert_writeable_eq!(&chars[i], s);
236 for j in 0..chars.len() {
237 assert_eq!(
238 crate::cmp_str(&chars[j], &s),
239 chars[j].cmp(&chars[i]),
240 "{:?} vs {:?}",
241 chars[j],
242 chars[i]
243 );
244 }
245 }
246
247 let arr: &[Cow<str>] = &[Cow::Borrowed(""), Cow::Owned("abc".to_string())];
249 check_writeable_slice(arr);
250
251 let arr: &[Box<str>] = &["".into(), "abc".into()];
253 check_writeable_slice(arr);
254
255 let arr: &[alloc::rc::Rc<str>] = &["".into(), "abc".into()];
257 check_writeable_slice(arr);
258
259 let arr: &[alloc::sync::Arc<str>] = &["".into(), "abc".into()];
261 check_writeable_slice(arr);
262
263 let arr: &[&String] = &[&String::new(), &"abc".to_owned()];
265 check_writeable_slice(arr);
266}