@@ -13,28 +13,48 @@ use crate::ffi::recovery as ffi;
13
13
use crate :: { key, Error , Message , Secp256k1 , Signing , Verification } ;
14
14
15
15
/// A tag used for recovering the public key from a compact signature.
16
- #[ derive( Copy , Clone , PartialEq , Eq , Debug , Ord , PartialOrd ) ]
17
- pub struct RecoveryId ( i32 ) ;
18
-
19
- /// An ECDSA signature with a recovery ID for pubkey recovery.
20
- #[ derive( Copy , Clone , PartialEq , Eq , Debug , Hash , Ord , PartialOrd ) ]
21
- pub struct RecoverableSignature ( ffi:: RecoverableSignature ) ;
16
+ #[ derive( Copy , Clone , PartialEq , Eq , Debug ) ]
17
+ pub enum RecoveryId {
18
+ /// Signature recovery ID 0
19
+ Zero ,
20
+ /// Signature recovery ID 1
21
+ One ,
22
+ /// Signature recovery ID 2
23
+ Two ,
24
+ /// Signature recovery ID 3
25
+ Three ,
26
+ }
22
27
23
- impl RecoveryId {
28
+ impl TryFrom < i32 > for RecoveryId {
29
+ type Error = Error ;
24
30
#[ inline]
25
- /// Allows library users to create valid recovery IDs from i32.
26
- pub fn from_i32 ( id : i32 ) -> Result < RecoveryId , Error > {
31
+ fn try_from ( id : i32 ) -> Result < RecoveryId , Error > {
27
32
match id {
28
- 0 ..=3 => Ok ( RecoveryId ( id) ) ,
33
+ 0 => Ok ( RecoveryId :: Zero ) ,
34
+ 1 => Ok ( RecoveryId :: One ) ,
35
+ 2 => Ok ( RecoveryId :: Two ) ,
36
+ 3 => Ok ( RecoveryId :: Three ) ,
29
37
_ => Err ( Error :: InvalidRecoveryId ) ,
30
38
}
31
39
}
40
+ }
32
41
42
+ impl From < RecoveryId > for i32 {
33
43
#[ inline]
34
- /// Allows library users to convert recovery IDs to i32.
35
- pub fn to_i32 ( self ) -> i32 { self . 0 }
44
+ fn from ( val : RecoveryId ) -> Self {
45
+ match val {
46
+ RecoveryId :: Zero => 0 ,
47
+ RecoveryId :: One => 1 ,
48
+ RecoveryId :: Two => 2 ,
49
+ RecoveryId :: Three => 3 ,
50
+ }
51
+ }
36
52
}
37
53
54
+ /// An ECDSA signature with a recovery ID for pubkey recovery.
55
+ #[ derive( Copy , Clone , PartialEq , Eq , Debug , Hash , Ord , PartialOrd ) ]
56
+ pub struct RecoverableSignature ( ffi:: RecoverableSignature ) ;
57
+
38
58
impl RecoverableSignature {
39
59
#[ inline]
40
60
/// Converts a compact-encoded byte slice to a signature. This
@@ -53,7 +73,7 @@ impl RecoverableSignature {
53
73
super_ffi:: secp256k1_context_no_precomp,
54
74
& mut ret,
55
75
data. as_c_ptr ( ) ,
56
- recid. 0 ,
76
+ recid. into ( ) ,
57
77
) == 1
58
78
{
59
79
Ok ( RecoverableSignature ( ret) )
@@ -80,7 +100,7 @@ impl RecoverableSignature {
80
100
/// Serializes the recoverable signature in compact format.
81
101
pub fn serialize_compact ( & self ) -> ( RecoveryId , [ u8 ; 64 ] ) {
82
102
let mut ret = [ 0u8 ; 64 ] ;
83
- let mut recid = 0i32 ;
103
+ let mut recid = RecoveryId :: Zero . into ( ) ;
84
104
unsafe {
85
105
let err = ffi:: secp256k1_ecdsa_recoverable_signature_serialize_compact (
86
106
super_ffi:: secp256k1_context_no_precomp,
@@ -90,7 +110,7 @@ impl RecoverableSignature {
90
110
) ;
91
111
assert ! ( err == 1 ) ;
92
112
}
93
- ( RecoveryId ( recid) , ret)
113
+ ( recid. try_into ( ) . expect ( "ffi returned invalid RecoveryId!" ) , ret)
94
114
}
95
115
96
116
/// Converts a recoverable signature to a non-recoverable one (this is needed
@@ -245,7 +265,7 @@ mod tests {
245
265
246
266
#[ test]
247
267
fn recid_sanity_check ( ) {
248
- let one = RecoveryId ( 1 ) ;
268
+ let one = RecoveryId :: One ;
249
269
assert_eq ! ( one, one. clone( ) ) ;
250
270
}
251
271
@@ -271,7 +291,7 @@ mod tests {
271
291
0x80 , 0x12 , 0x0e , 0xf8 , 0x02 , 0x5e , 0x70 , 0x9f ,
272
292
0xff , 0x20 , 0x80 , 0xc4 , 0xa3 , 0x9a , 0xae , 0x06 ,
273
293
0x8d , 0x12 , 0xee , 0xd0 , 0x09 , 0xb6 , 0x8c , 0x89 ] ,
274
- RecoveryId ( 1 ) ) )
294
+ RecoveryId :: One ) )
275
295
}
276
296
277
297
#[ test]
@@ -297,7 +317,7 @@ mod tests {
297
317
0xd9 , 0xfd , 0xdb , 0x44 , 0xbd , 0x0d , 0xd9 , 0xb9 ,
298
318
0xdd , 0x47 , 0x66 , 0x6a , 0xb5 , 0x28 , 0x71 , 0x90 ,
299
319
0x1d , 0x17 , 0x61 , 0xeb , 0x82 , 0xec , 0x87 , 0x22 ] ,
300
- RecoveryId ( 0 ) ) )
320
+ RecoveryId :: Zero ) )
301
321
}
302
322
303
323
#[ test]
@@ -365,10 +385,10 @@ mod tests {
365
385
let msg = Message :: from_digest_slice ( & [ 0x55 ; 32 ] ) . unwrap ( ) ;
366
386
367
387
// Zero is not a valid sig
368
- let sig = RecoverableSignature :: from_compact ( & [ 0 ; 64 ] , RecoveryId ( 0 ) ) . unwrap ( ) ;
388
+ let sig = RecoverableSignature :: from_compact ( & [ 0 ; 64 ] , RecoveryId :: Zero ) . unwrap ( ) ;
369
389
assert_eq ! ( s. recover_ecdsa( & msg, & sig) , Err ( Error :: InvalidSignature ) ) ;
370
390
// ...but 111..111 is
371
- let sig = RecoverableSignature :: from_compact ( & [ 1 ; 64 ] , RecoveryId ( 0 ) ) . unwrap ( ) ;
391
+ let sig = RecoverableSignature :: from_compact ( & [ 1 ; 64 ] , RecoveryId :: Zero ) . unwrap ( ) ;
372
392
assert ! ( s. recover_ecdsa( & msg, & sig) . is_ok( ) ) ;
373
393
}
374
394
@@ -384,13 +404,13 @@ mod tests {
384
404
0x80 , 0x12 , 0x0e , 0xf8 , 0x02 , 0x5e , 0x70 , 0x9f ,
385
405
0xff , 0x20 , 0x80 , 0xc4 , 0xa3 , 0x9a , 0xae , 0x06 ,
386
406
0x8d , 0x12 , 0xee , 0xd0 , 0x09 , 0xb6 , 0x8c , 0x89 ] ,
387
- RecoveryId ( 1 ) ) . unwrap ( ) ;
407
+ RecoveryId :: One ) . unwrap ( ) ;
388
408
assert_eq ! ( & format!( "{:?}" , sig) , "RecoverableSignature(6673ffad2147741f04772b6f921f0ba6af0c1e77fc439e65c36dedf4092e88984c1a971652e0ada880120ef8025e709fff2080c4a39aae068d12eed009b68c8901)" ) ;
389
409
}
390
410
391
411
#[ test]
392
412
fn test_recov_sig_serialize_compact ( ) {
393
- let recid_in = RecoveryId ( 1 ) ;
413
+ let recid_in = RecoveryId :: One ;
394
414
#[ rustfmt:: skip]
395
415
let bytes_in = & [
396
416
0x66 , 0x73 , 0xff , 0xad , 0x21 , 0x47 , 0x74 , 0x1f ,
@@ -409,16 +429,16 @@ mod tests {
409
429
410
430
#[ test]
411
431
fn test_recov_id_conversion_between_i32 ( ) {
412
- assert ! ( RecoveryId :: from_i32 ( - 1 ) . is_err( ) ) ;
413
- assert ! ( RecoveryId :: from_i32 ( 0 ) . is_ok( ) ) ;
414
- assert ! ( RecoveryId :: from_i32 ( 1 ) . is_ok( ) ) ;
415
- assert ! ( RecoveryId :: from_i32 ( 2 ) . is_ok( ) ) ;
416
- assert ! ( RecoveryId :: from_i32 ( 3 ) . is_ok( ) ) ;
417
- assert ! ( RecoveryId :: from_i32 ( 4 ) . is_err( ) ) ;
418
- let id0 = RecoveryId :: from_i32 ( 0 ) . unwrap ( ) ;
419
- assert_eq ! ( id0 . to_i32 ( ) , 0 ) ;
420
- let id1 = RecoveryId ( 1 ) ;
421
- assert_eq ! ( id1 . to_i32 ( ) , 1 ) ;
432
+ assert ! ( RecoveryId :: try_from ( - 1i32 ) . is_err( ) ) ;
433
+ assert ! ( RecoveryId :: try_from ( 0i32 ) . is_ok( ) ) ;
434
+ assert ! ( RecoveryId :: try_from ( 1i32 ) . is_ok( ) ) ;
435
+ assert ! ( RecoveryId :: try_from ( 2i32 ) . is_ok( ) ) ;
436
+ assert ! ( RecoveryId :: try_from ( 3i32 ) . is_ok( ) ) ;
437
+ assert ! ( RecoveryId :: try_from ( 4i32 ) . is_err( ) ) ;
438
+ let id0 = RecoveryId :: Zero ;
439
+ assert_eq ! ( Into :: < i32 > :: into ( id0 ) , 0i32 ) ;
440
+ let id1 = RecoveryId :: One ;
441
+ assert_eq ! ( Into :: < i32 > :: into ( id1 ) , 1i32 ) ;
422
442
}
423
443
}
424
444
0 commit comments