@@ -74,6 +74,8 @@ use core::{fmt, mem};
74
74
#[ cfg( feature = "std" ) ]
75
75
use std:: borrow:: Cow ;
76
76
77
+ mod error;
78
+
77
79
/// Integer in the range `0..32`
78
80
#[ derive( PartialEq , Eq , Debug , Copy , Clone , Default , PartialOrd , Ord , Hash ) ]
79
81
#[ allow( non_camel_case_types) ]
@@ -91,18 +93,56 @@ impl From<u5> for u8 {
91
93
fn from ( v : u5 ) -> u8 { v. 0 }
92
94
}
93
95
94
- impl TryFrom < u8 > for u5 {
95
- type Error = Error ;
96
-
97
- /// Errors if `value` is out of range.
98
- fn try_from ( value : u8 ) -> Result < Self , Self :: Error > {
99
- if value > 31 {
100
- Err ( Error :: InvalidData ( value) )
101
- } else {
102
- Ok ( u5 ( value) )
103
- }
96
+ macro_rules! impl_try_from_upper_bounded {
97
+ ( $( $ty: ident) +) => {
98
+ $(
99
+ impl TryFrom <$ty> for u5 {
100
+ type Error = TryFromIntError ;
101
+
102
+ /// Tries to create the target number type from a source number type.
103
+ ///
104
+ /// # Errors
105
+ ///
106
+ /// Returns an error if `value` overflows a `u5`.
107
+ fn try_from( value: $ty) -> Result <Self , Self :: Error > {
108
+ if value > 31 {
109
+ Err ( TryFromIntError :: PosOverflow )
110
+ } else {
111
+ let x = u8 :: try_from( value) . expect( "within range" ) ;
112
+ Ok ( u5( x) )
113
+ }
114
+ }
115
+ }
116
+ ) +
104
117
}
105
118
}
119
+ macro_rules! impl_try_from_both_bounded {
120
+ ( $( $ty: ident) +) => {
121
+ $(
122
+ impl TryFrom <$ty> for u5 {
123
+ type Error = TryFromIntError ;
124
+
125
+ /// Tries to create the target number type from a source number type.
126
+ ///
127
+ /// # Errors
128
+ ///
129
+ /// Returns an error if `value` is outside of the range of a `u5`.
130
+ fn try_from( value: $ty) -> Result <Self , Self :: Error > {
131
+ if value < 0 {
132
+ Err ( TryFromIntError :: NegOverflow )
133
+ } else if value > 31 {
134
+ Err ( TryFromIntError :: PosOverflow )
135
+ } else {
136
+ let x = u8 :: try_from( value) . expect( "within range" ) ;
137
+ Ok ( u5( x) )
138
+ }
139
+ }
140
+ }
141
+ ) +
142
+ }
143
+ }
144
+ impl_try_from_upper_bounded ! ( u8 u16 u32 u64 u128 ) ;
145
+ impl_try_from_both_bounded ! ( i8 i16 i32 i64 i128 ) ;
106
146
107
147
impl AsRef < u8 > for u5 {
108
148
fn as_ref ( & self ) -> & u8 { & self . 0 }
@@ -340,7 +380,10 @@ impl<T: AsRef<[u8]>> CheckBase32<Vec<u5>> for T {
340
380
type Err = Error ;
341
381
342
382
fn check_base32 ( self ) -> Result < Vec < u5 > , Self :: Err > {
343
- self . as_ref ( ) . iter ( ) . map ( |x| u5:: try_from ( * x) ) . collect :: < Result < Vec < u5 > , Error > > ( )
383
+ self . as_ref ( )
384
+ . iter ( )
385
+ . map ( |x| u5:: try_from ( * x) . map_err ( Error :: TryFrom ) )
386
+ . collect :: < Result < Vec < u5 > , Error > > ( )
344
387
}
345
388
}
346
389
@@ -669,12 +712,14 @@ pub enum Error {
669
712
InvalidLength ,
670
713
/// Some part of the string contains an invalid character.
671
714
InvalidChar ( char ) ,
672
- /// Some part of the data has an invalid value.
673
- InvalidData ( u8 ) ,
674
715
/// The bit conversion failed due to a padding issue.
675
716
InvalidPadding ,
676
717
/// The whole string must be of one case.
677
718
MixedCase ,
719
+ /// Attempted to convert a value which overflows a `u5`.
720
+ Overflow ,
721
+ /// Conversion to u5 failed.
722
+ TryFrom ( TryFromIntError ) ,
678
723
}
679
724
680
725
impl fmt:: Display for Error {
@@ -686,9 +731,10 @@ impl fmt::Display for Error {
686
731
InvalidChecksum => write ! ( f, "invalid checksum" ) ,
687
732
InvalidLength => write ! ( f, "invalid length" ) ,
688
733
InvalidChar ( n) => write ! ( f, "invalid character (code={})" , n) ,
689
- InvalidData ( n) => write ! ( f, "invalid data point ({})" , n) ,
690
734
InvalidPadding => write ! ( f, "invalid padding" ) ,
691
735
MixedCase => write ! ( f, "mixed-case strings not allowed" ) ,
736
+ TryFrom ( ref e) => write_err ! ( f, "conversion to u5 failed" ; e) ,
737
+ Overflow => write ! ( f, "attempted to convert a value which overflows a u5" ) ,
692
738
}
693
739
}
694
740
}
@@ -699,12 +745,54 @@ impl std::error::Error for Error {
699
745
use Error :: * ;
700
746
701
747
match * self {
748
+ TryFrom ( ref e) => Some ( e) ,
702
749
MissingSeparator | InvalidChecksum | InvalidLength | InvalidChar ( _)
703
- | InvalidData ( _ ) | InvalidPadding | MixedCase => None ,
750
+ | InvalidPadding | MixedCase | Overflow => None ,
704
751
}
705
752
}
706
753
}
707
754
755
+ impl From < TryFromIntError > for Error {
756
+ fn from ( e : TryFromIntError ) -> Self { Error :: TryFrom ( e) }
757
+ }
758
+
759
+ /// Error return when TryFrom<T> fails for T -> u5 conversion.
760
+ #[ derive( Copy , Clone , PartialEq , Eq , PartialOrd , Ord , Hash , Debug ) ]
761
+ pub enum TryFromIntError {
762
+ /// Attempted to convert a negative value to a `u5`.
763
+ NegOverflow ,
764
+ /// Attempted to convert a value which overflows a `u5`.
765
+ PosOverflow ,
766
+ }
767
+
768
+ impl fmt:: Display for TryFromIntError {
769
+ fn fmt ( & self , f : & mut fmt:: Formatter < ' _ > ) -> fmt:: Result {
770
+ use TryFromIntError :: * ;
771
+
772
+ match * self {
773
+ NegOverflow => write ! ( f, "attempted to convert a negative value to a u5" ) ,
774
+ PosOverflow => write ! ( f, "attempted to convert a value which overflows a u5" ) ,
775
+ }
776
+ }
777
+ }
778
+
779
+ #[ cfg( feature = "std" ) ]
780
+ impl std:: error:: Error for TryFromIntError {
781
+ fn source ( & self ) -> Option < & ( dyn std:: error:: Error + ' static ) > {
782
+ use TryFromIntError :: * ;
783
+
784
+ match * self {
785
+ NegOverflow | PosOverflow => None ,
786
+ }
787
+ }
788
+ }
789
+
790
+ // impl From<convert::Error> for Error {
791
+ // fn from(e: convert::Error) -> Self {
792
+ // Error::InvalidData(e)
793
+ // }
794
+ // }
795
+
708
796
/// Convert between bit sizes
709
797
///
710
798
/// # Errors
@@ -738,7 +826,7 @@ where
738
826
let v: u32 = u32:: from ( Into :: < u8 > :: into ( * value) ) ;
739
827
if ( v >> from) != 0 {
740
828
// Input value exceeds `from` bit size
741
- return Err ( Error :: InvalidData ( v as u8 ) ) ;
829
+ return Err ( Error :: Overflow ) ;
742
830
}
743
831
acc = ( acc << from) | v;
744
832
bits += from;
@@ -885,7 +973,7 @@ mod tests {
885
973
// Set of [data, from_bits, to_bits, pad, expected error]
886
974
let tests: Vec < ( Vec < u8 > , u32 , u32 , bool , Error ) > = vec ! [
887
975
( vec![ 0xff ] , 8 , 5 , false , Error :: InvalidPadding ) ,
888
- ( vec![ 0x02 ] , 1 , 1 , true , Error :: InvalidData ( 0x02 ) ) ,
976
+ ( vec![ 0x02 ] , 1 , 1 , true , Error :: Overflow ) ,
889
977
] ;
890
978
for t in tests {
891
979
let ( data, from_bits, to_bits, pad, expected_error) = t;
@@ -918,7 +1006,7 @@ mod tests {
918
1006
assert ! ( [ 0u8 , 1 , 2 , 30 , 31 , 255 ] . check_base32( ) . is_err( ) ) ;
919
1007
920
1008
assert ! ( [ 1u8 , 2 , 3 , 4 ] . check_base32( ) . is_ok( ) ) ;
921
- assert_eq ! ( [ 30u8 , 31 , 35 , 20 ] . check_base32( ) , Err ( Error :: InvalidData ( 35 ) ) ) ;
1009
+ assert_eq ! ( [ 30u8 , 31 , 35 , 20 ] . check_base32( ) , Err ( Error :: TryFrom ( TryFromIntError :: PosOverflow ) ) )
922
1010
}
923
1011
924
1012
#[ test]
@@ -1028,4 +1116,13 @@ mod tests {
1028
1116
1029
1117
assert_eq ! ( encoded_str, "hrp1qqqq40atq3" ) ;
1030
1118
}
1119
+
1120
+ #[ test]
1121
+ fn try_from_err ( ) {
1122
+ assert ! ( u5:: try_from( 32_u8 ) . is_err( ) ) ;
1123
+ assert ! ( u5:: try_from( 32_u16 ) . is_err( ) ) ;
1124
+ assert ! ( u5:: try_from( 32_u32 ) . is_err( ) ) ;
1125
+ assert ! ( u5:: try_from( 32_u64 ) . is_err( ) ) ;
1126
+ assert ! ( u5:: try_from( 32_u128 ) . is_err( ) ) ;
1127
+ }
1031
1128
}
0 commit comments