1use core::convert::TryFrom;
2use core::marker::PhantomData;
3
4use crate::__private::maybestd::{
5 borrow::{Cow, ToOwned},
6 boxed::Box,
7 collections::{BTreeMap, BTreeSet, LinkedList, VecDeque},
8 string::String,
9 vec::Vec,
10};
11use crate::error::check_zst;
12use crate::io::{Error, ErrorKind, Result, Write};
13
14pub(crate) mod helpers;
15
16const FLOAT_NAN_ERR: &str = "For portability reasons we do not allow to serialize NaNs.";
17
18pub trait BorshSerialize {
52 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()>;
53
54 #[inline]
55 #[doc(hidden)]
56 fn u8_slice(slice: &[Self]) -> Option<&[u8]>
57 where
58 Self: Sized,
59 {
60 let _ = slice;
61 None
62 }
63}
64
65impl BorshSerialize for u8 {
66 #[inline]
67 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
68 writer.write_all(core::slice::from_ref(self))
69 }
70
71 #[inline]
72 fn u8_slice(slice: &[Self]) -> Option<&[u8]> {
73 Some(slice)
74 }
75}
76
77macro_rules! impl_for_integer {
78 ($type: ident) => {
79 impl BorshSerialize for $type {
80 #[inline]
81 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
82 let bytes = self.to_le_bytes();
83 writer.write_all(&bytes)
84 }
85 }
86 };
87}
88
89impl_for_integer!(i8);
90impl_for_integer!(i16);
91impl_for_integer!(i32);
92impl_for_integer!(i64);
93impl_for_integer!(i128);
94impl_for_integer!(u16);
95impl_for_integer!(u32);
96impl_for_integer!(u64);
97impl_for_integer!(u128);
98
99macro_rules! impl_for_nonzero_integer {
100 ($type: ty) => {
101 impl BorshSerialize for $type {
102 #[inline]
103 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
104 BorshSerialize::serialize(&self.get(), writer)
105 }
106 }
107 };
108}
109
110impl_for_nonzero_integer!(core::num::NonZeroI8);
111impl_for_nonzero_integer!(core::num::NonZeroI16);
112impl_for_nonzero_integer!(core::num::NonZeroI32);
113impl_for_nonzero_integer!(core::num::NonZeroI64);
114impl_for_nonzero_integer!(core::num::NonZeroI128);
115impl_for_nonzero_integer!(core::num::NonZeroU8);
116impl_for_nonzero_integer!(core::num::NonZeroU16);
117impl_for_nonzero_integer!(core::num::NonZeroU32);
118impl_for_nonzero_integer!(core::num::NonZeroU64);
119impl_for_nonzero_integer!(core::num::NonZeroU128);
120impl_for_nonzero_integer!(core::num::NonZeroUsize);
121
122impl BorshSerialize for isize {
123 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
124 BorshSerialize::serialize(&(*self as i64), writer)
125 }
126}
127
128impl BorshSerialize for usize {
129 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
130 BorshSerialize::serialize(&(*self as u64), writer)
131 }
132}
133
134macro_rules! impl_for_float {
137 ($type: ident) => {
138 impl BorshSerialize for $type {
139 #[inline]
140 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
141 if self.is_nan() {
142 return Err(Error::new(ErrorKind::InvalidData, FLOAT_NAN_ERR));
143 }
144 writer.write_all(&self.to_bits().to_le_bytes())
145 }
146 }
147 };
148}
149
150impl_for_float!(f32);
151impl_for_float!(f64);
152
153impl BorshSerialize for bool {
154 #[inline]
155 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
156 (u8::from(*self)).serialize(writer)
157 }
158}
159
160impl<T> BorshSerialize for Option<T>
161where
162 T: BorshSerialize,
163{
164 #[inline]
165 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
166 match self {
167 None => 0u8.serialize(writer),
168 Some(value) => {
169 1u8.serialize(writer)?;
170 value.serialize(writer)
171 }
172 }
173 }
174}
175
176impl<T, E> BorshSerialize for core::result::Result<T, E>
177where
178 T: BorshSerialize,
179 E: BorshSerialize,
180{
181 #[inline]
182 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
183 match self {
184 Err(e) => {
185 0u8.serialize(writer)?;
186 e.serialize(writer)
187 }
188 Ok(v) => {
189 1u8.serialize(writer)?;
190 v.serialize(writer)
191 }
192 }
193 }
194}
195
196impl BorshSerialize for str {
197 #[inline]
198 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
199 self.as_bytes().serialize(writer)
200 }
201}
202
203impl BorshSerialize for String {
204 #[inline]
205 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
206 self.as_bytes().serialize(writer)
207 }
208}
209
210#[cfg(feature = "ascii")]
212pub mod ascii {
213 use super::BorshSerialize;
217 use crate::io::{Result, Write};
218
219 impl BorshSerialize for ascii::AsciiChar {
220 #[inline]
221 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
222 self.as_byte().serialize(writer)
223 }
224 }
225
226 impl BorshSerialize for ascii::AsciiStr {
227 #[inline]
228 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
229 self.as_bytes().serialize(writer)
230 }
231 }
232
233 impl BorshSerialize for ascii::AsciiString {
234 #[inline]
235 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
236 self.as_bytes().serialize(writer)
237 }
238 }
239}
240
241#[inline]
243fn serialize_slice<T: BorshSerialize, W: Write>(data: &[T], writer: &mut W) -> Result<()> {
244 if let Some(u8_slice) = T::u8_slice(data) {
245 writer.write_all(u8_slice)?;
246 } else {
247 for item in data {
248 item.serialize(writer)?;
249 }
250 }
251 Ok(())
252}
253
254impl<T> BorshSerialize for [T]
255where
256 T: BorshSerialize,
257{
258 #[inline]
259 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
260 writer.write_all(
261 &(u32::try_from(self.len()).map_err(|_| ErrorKind::InvalidData)?).to_le_bytes(),
262 )?;
263 serialize_slice(self, writer)
264 }
265}
266
267impl<T: BorshSerialize + ?Sized> BorshSerialize for &T {
268 #[inline]
269 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
270 (*self).serialize(writer)
271 }
272}
273
274impl<T> BorshSerialize for Cow<'_, T>
275where
276 T: BorshSerialize + ToOwned + ?Sized,
277{
278 #[inline]
279 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
280 self.as_ref().serialize(writer)
281 }
282}
283
284impl<T> BorshSerialize for Vec<T>
285where
286 T: BorshSerialize,
287{
288 #[inline]
289 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
290 check_zst::<T>()?;
291
292 self.as_slice().serialize(writer)
293 }
294}
295
296#[cfg(feature = "bytes")]
297impl BorshSerialize for bytes::Bytes {
298 #[inline]
299 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
300 self.as_ref().serialize(writer)
301 }
302}
303
304#[cfg(feature = "bytes")]
305impl BorshSerialize for bytes::BytesMut {
306 #[inline]
307 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
308 self.as_ref().serialize(writer)
309 }
310}
311
312#[cfg(feature = "bson")]
313impl BorshSerialize for bson::oid::ObjectId {
314 #[inline]
315 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
316 self.bytes().serialize(writer)
317 }
318}
319
320#[cfg(feature = "indexmap")]
321impl<T, S> BorshSerialize for indexmap::IndexSet<T, S>
324where
325 T: BorshSerialize,
326{
327 #[inline]
328 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
329 check_zst::<T>()?;
330
331 let iterator = self.iter();
332
333 u32::try_from(iterator.len())
334 .map_err(|_| ErrorKind::InvalidData)?
335 .serialize(writer)?;
336
337 for item in iterator {
338 item.serialize(writer)?;
339 }
340
341 Ok(())
342 }
343}
344
345#[cfg(feature = "indexmap")]
346impl<K, V, S> BorshSerialize for indexmap::IndexMap<K, V, S>
349where
350 K: BorshSerialize,
351 V: BorshSerialize,
352{
353 #[inline]
354 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
355 check_zst::<K>()?;
356
357 let iterator = self.iter();
358
359 u32::try_from(iterator.len())
360 .map_err(|_| ErrorKind::InvalidData)?
361 .serialize(writer)?;
362
363 for (key, value) in iterator {
364 key.serialize(writer)?;
365 value.serialize(writer)?;
366 }
367
368 Ok(())
369 }
370}
371
372impl<T> BorshSerialize for VecDeque<T>
373where
374 T: BorshSerialize,
375{
376 #[inline]
377 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
378 check_zst::<T>()?;
379
380 writer.write_all(
381 &(u32::try_from(self.len()).map_err(|_| ErrorKind::InvalidData)?).to_le_bytes(),
382 )?;
383 let slices = self.as_slices();
384 serialize_slice(slices.0, writer)?;
385 serialize_slice(slices.1, writer)
386 }
387}
388
389impl<T> BorshSerialize for LinkedList<T>
390where
391 T: BorshSerialize,
392{
393 #[inline]
394 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
395 check_zst::<T>()?;
396
397 writer.write_all(
398 &(u32::try_from(self.len()).map_err(|_| ErrorKind::InvalidData)?).to_le_bytes(),
399 )?;
400 for item in self {
401 item.serialize(writer)?;
402 }
403 Ok(())
404 }
405}
406
407#[cfg(hash_collections)]
412pub mod hashes {
413 use crate::__private::maybestd::vec::Vec;
414 use crate::error::check_zst;
415 use crate::{
416 BorshSerialize,
417 __private::maybestd::collections::{HashMap, HashSet},
418 };
419 use core::convert::TryFrom;
420 use core::hash::BuildHasher;
421
422 use crate::io::{ErrorKind, Result, Write};
423
424 impl<K, V, H> BorshSerialize for HashMap<K, V, H>
425 where
426 K: BorshSerialize + Ord,
427 V: BorshSerialize,
428 H: BuildHasher,
429 {
430 #[inline]
431 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
432 check_zst::<K>()?;
433
434 let mut vec = self.iter().collect::<Vec<_>>();
435 vec.sort_by(|(a, _), (b, _)| a.cmp(b));
436 u32::try_from(vec.len())
437 .map_err(|_| ErrorKind::InvalidData)?
438 .serialize(writer)?;
439 for kv in vec {
440 kv.serialize(writer)?;
441 }
442 Ok(())
443 }
444 }
445
446 impl<T, H> BorshSerialize for HashSet<T, H>
447 where
448 T: BorshSerialize + Ord,
449 H: BuildHasher,
450 {
451 #[inline]
452 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
453 check_zst::<T>()?;
454
455 let mut vec = self.iter().collect::<Vec<_>>();
456 vec.sort();
457 u32::try_from(vec.len())
458 .map_err(|_| ErrorKind::InvalidData)?
459 .serialize(writer)?;
460 for item in vec {
461 item.serialize(writer)?;
462 }
463 Ok(())
464 }
465 }
466}
467
468impl<K, V> BorshSerialize for BTreeMap<K, V>
469where
470 K: BorshSerialize,
471 V: BorshSerialize,
472{
473 #[inline]
474 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
475 check_zst::<K>()?;
476 u32::try_from(self.len())
480 .map_err(|_| ErrorKind::InvalidData)?
481 .serialize(writer)?;
482 for (key, value) in self {
483 key.serialize(writer)?;
484 value.serialize(writer)?;
485 }
486 Ok(())
487 }
488}
489
490impl<T> BorshSerialize for BTreeSet<T>
491where
492 T: BorshSerialize,
493{
494 #[inline]
495 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
496 check_zst::<T>()?;
497 u32::try_from(self.len())
500 .map_err(|_| ErrorKind::InvalidData)?
501 .serialize(writer)?;
502 for item in self {
503 item.serialize(writer)?;
504 }
505 Ok(())
506 }
507}
508
509#[cfg(feature = "std")]
510impl BorshSerialize for std::net::SocketAddr {
511 #[inline]
512 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
513 match *self {
514 std::net::SocketAddr::V4(ref addr) => {
515 0u8.serialize(writer)?;
516 addr.serialize(writer)
517 }
518 std::net::SocketAddr::V6(ref addr) => {
519 1u8.serialize(writer)?;
520 addr.serialize(writer)
521 }
522 }
523 }
524}
525
526#[cfg(feature = "std")]
527impl BorshSerialize for std::net::SocketAddrV4 {
528 #[inline]
529 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
530 self.ip().serialize(writer)?;
531 self.port().serialize(writer)
532 }
533}
534
535#[cfg(feature = "std")]
536impl BorshSerialize for std::net::SocketAddrV6 {
537 #[inline]
538 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
539 self.ip().serialize(writer)?;
540 self.port().serialize(writer)
541 }
542}
543
544#[cfg(feature = "std")]
545impl BorshSerialize for std::net::Ipv4Addr {
546 #[inline]
547 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
548 writer.write_all(&self.octets())
549 }
550}
551
552#[cfg(feature = "std")]
553impl BorshSerialize for std::net::Ipv6Addr {
554 #[inline]
555 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
556 writer.write_all(&self.octets())
557 }
558}
559
560#[cfg(feature = "std")]
561impl BorshSerialize for std::net::IpAddr {
562 #[inline]
563 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
564 match self {
565 std::net::IpAddr::V4(ipv4) => {
566 writer.write_all(&0u8.to_le_bytes())?;
567 ipv4.serialize(writer)
568 }
569 std::net::IpAddr::V6(ipv6) => {
570 writer.write_all(&1u8.to_le_bytes())?;
571 ipv6.serialize(writer)
572 }
573 }
574 }
575}
576impl<T: BorshSerialize + ?Sized> BorshSerialize for Box<T> {
577 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
578 self.as_ref().serialize(writer)
579 }
580}
581
582impl<T, const N: usize> BorshSerialize for [T; N]
583where
584 T: BorshSerialize,
585{
586 #[inline]
587 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
588 if N == 0 {
589 return Ok(());
590 } else if let Some(u8_slice) = T::u8_slice(self) {
591 writer.write_all(u8_slice)?;
592 } else {
593 for el in self.iter() {
594 el.serialize(writer)?;
595 }
596 }
597 Ok(())
598 }
599}
600
601macro_rules! impl_tuple {
602 (@unit $name:ty) => {
603 impl BorshSerialize for $name {
604 #[inline]
605 fn serialize<W: Write>(&self, _writer: &mut W) -> Result<()> {
606 Ok(())
607 }
608 }
609 };
610
611 ($($idx:tt $name:ident)+) => {
612 impl<$($name),+> BorshSerialize for ($($name,)+)
613 where $($name: BorshSerialize,)+
614 {
615 #[inline]
616 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
617 $(self.$idx.serialize(writer)?;)+
618 Ok(())
619 }
620 }
621 };
622}
623
624impl_tuple!(@unit ());
625impl_tuple!(@unit core::ops::RangeFull);
626
627impl_tuple!(0 T0);
628impl_tuple!(0 T0 1 T1);
629impl_tuple!(0 T0 1 T1 2 T2);
630impl_tuple!(0 T0 1 T1 2 T2 3 T3);
631impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4);
632impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4 5 T5);
633impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4 5 T5 6 T6);
634impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4 5 T5 6 T6 7 T7);
635impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4 5 T5 6 T6 7 T7 8 T8);
636impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4 5 T5 6 T6 7 T7 8 T8 9 T9);
637impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4 5 T5 6 T6 7 T7 8 T8 9 T9 10 T10);
638impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4 5 T5 6 T6 7 T7 8 T8 9 T9 10 T10 11 T11);
639impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4 5 T5 6 T6 7 T7 8 T8 9 T9 10 T10 11 T11 12 T12);
640impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4 5 T5 6 T6 7 T7 8 T8 9 T9 10 T10 11 T11 12 T12 13 T13);
641impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4 5 T5 6 T6 7 T7 8 T8 9 T9 10 T10 11 T11 12 T12 13 T13 14 T14);
642impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4 5 T5 6 T6 7 T7 8 T8 9 T9 10 T10 11 T11 12 T12 13 T13 14 T14 15 T15);
643impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4 5 T5 6 T6 7 T7 8 T8 9 T9 10 T10 11 T11 12 T12 13 T13 14 T14 15 T15 16 T16);
644impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4 5 T5 6 T6 7 T7 8 T8 9 T9 10 T10 11 T11 12 T12 13 T13 14 T14 15 T15 16 T16 17 T17);
645impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4 5 T5 6 T6 7 T7 8 T8 9 T9 10 T10 11 T11 12 T12 13 T13 14 T14 15 T15 16 T16 17 T17 18 T18);
646impl_tuple!(0 T0 1 T1 2 T2 3 T3 4 T4 5 T5 6 T6 7 T7 8 T8 9 T9 10 T10 11 T11 12 T12 13 T13 14 T14 15 T15 16 T16 17 T17 18 T18 19 T19);
647
648macro_rules! impl_range {
649 ($type:ident, $this:ident, $($field:expr),*) => {
650 impl<T: BorshSerialize> BorshSerialize for core::ops::$type<T> {
651 #[inline]
652 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
653 let $this = self;
654 $( $field.serialize(writer)?; )*
655 Ok(())
656 }
657 }
658 };
659}
660
661impl_range!(Range, this, &this.start, &this.end);
662impl_range!(RangeInclusive, this, this.start(), this.end());
663impl_range!(RangeFrom, this, &this.start);
664impl_range!(RangeTo, this, &this.end);
665impl_range!(RangeToInclusive, this, &this.end);
666
667#[cfg(feature = "rc")]
669pub mod rc {
670 use crate::__private::maybestd::{rc::Rc, sync::Arc};
674 use crate::io::{Result, Write};
675 use crate::BorshSerialize;
676
677 impl<T: BorshSerialize + ?Sized> BorshSerialize for Rc<T> {
684 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
685 (**self).serialize(writer)
686 }
687 }
688
689 impl<T: BorshSerialize + ?Sized> BorshSerialize for Arc<T> {
696 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
697 (**self).serialize(writer)
698 }
699 }
700}
701
702impl<T: ?Sized> BorshSerialize for PhantomData<T> {
703 fn serialize<W: Write>(&self, _: &mut W) -> Result<()> {
704 Ok(())
705 }
706}
707
708impl<T> BorshSerialize for core::cell::Cell<T>
709where
710 T: BorshSerialize + Copy,
711{
712 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
713 <T as BorshSerialize>::serialize(&self.get(), writer)
714 }
715}
716
717impl<T> BorshSerialize for core::cell::RefCell<T>
718where
719 T: BorshSerialize + Sized,
720{
721 fn serialize<W: Write>(&self, writer: &mut W) -> Result<()> {
722 match self.try_borrow() {
723 Ok(ref value) => value.serialize(writer),
724 Err(_) => Err(Error::new(ErrorKind::Other, "already mutably borrowed")),
725 }
726 }
727}