@@ -182,6 +182,72 @@ impl PublicKey {
182
182
buf
183
183
}
184
184
185
+ /// Serialize the public key into a `SortKey`.
186
+ ///
187
+ /// `SortKey` is not too useful by itself, but it can be used to sort a
188
+ /// `[PublicKey]` slice using `sort_unstable_by_key`, `sort_by_cached_key`,
189
+ /// `sort_by_key`, or any of the other `*_by_key` methods on slice.
190
+ /// Pass the method into the sort method directly. (ie. `PublicKey::to_sort_key`)
191
+ ///
192
+ /// This method of sorting is in line with Bitcoin Core's implementation of
193
+ /// sorting keys for output descriptors such as `sortedmulti()`.
194
+ ///
195
+ /// If every `PublicKey` in the slice is `compressed == true` then this will sort
196
+ /// the keys in a
197
+ /// [BIP67](https://github.com/bitcoin/bips/blob/master/bip-0067.mediawiki)
198
+ /// compliant way.
199
+ ///
200
+ /// # Example: Using with `sort_unstable_by_key`
201
+ ///
202
+ /// ```rust
203
+ /// use std::str::FromStr;
204
+ /// use bitcoin::PublicKey;
205
+ ///
206
+ /// let pk = |s| PublicKey::from_str(s).unwrap();
207
+ ///
208
+ /// let mut unsorted = [
209
+ /// pk("04c4b0bbb339aa236bff38dbe6a451e111972a7909a126bc424013cba2ec33bc38e98ac269ffe028345c31ac8d0a365f29c8f7e7cfccac72f84e1acd02bc554f35"),
210
+ /// pk("038f47dcd43ba6d97fc9ed2e3bba09b175a45fac55f0683e8cf771e8ced4572354"),
211
+ /// pk("028bde91b10013e08949a318018fedbd896534a549a278e220169ee2a36517c7aa"),
212
+ /// pk("04c4b0bbb339aa236bff38dbe6a451e111972a7909a126bc424013cba2ec33bc3816753d96001fd7cba3ce5372f5c9a0d63708183033538d07b1e532fc43aaacfa"),
213
+ /// pk("032b8324c93575034047a52e9bca05a46d8347046b91a032eff07d5de8d3f2730b"),
214
+ /// pk("045d753414fa292ea5b8f56e39cfb6a0287b2546231a5cb05c4b14ab4b463d171f5128148985b23eccb1e2905374873b1f09b9487f47afa6b1f2b0083ac8b4f7e8"),
215
+ /// pk("0234dd69c56c36a41230d573d68adeae0030c9bc0bf26f24d3e1b64c604d293c68"),
216
+ /// ];
217
+ /// let sorted = [
218
+ /// // These first 4 keys are in a BIP67 compatible sorted order
219
+ /// // (since they are compressed)
220
+ /// pk("0234dd69c56c36a41230d573d68adeae0030c9bc0bf26f24d3e1b64c604d293c68"),
221
+ /// pk("028bde91b10013e08949a318018fedbd896534a549a278e220169ee2a36517c7aa"),
222
+ /// pk("032b8324c93575034047a52e9bca05a46d8347046b91a032eff07d5de8d3f2730b"),
223
+ /// pk("038f47dcd43ba6d97fc9ed2e3bba09b175a45fac55f0683e8cf771e8ced4572354"),
224
+ /// // Uncompressed keys are not BIP67 compliant, but are sorted
225
+ /// // after compressed keys in Bitcoin Core using `sortedmulti()`
226
+ /// pk("045d753414fa292ea5b8f56e39cfb6a0287b2546231a5cb05c4b14ab4b463d171f5128148985b23eccb1e2905374873b1f09b9487f47afa6b1f2b0083ac8b4f7e8"),
227
+ /// pk("04c4b0bbb339aa236bff38dbe6a451e111972a7909a126bc424013cba2ec33bc3816753d96001fd7cba3ce5372f5c9a0d63708183033538d07b1e532fc43aaacfa"),
228
+ /// pk("04c4b0bbb339aa236bff38dbe6a451e111972a7909a126bc424013cba2ec33bc38e98ac269ffe028345c31ac8d0a365f29c8f7e7cfccac72f84e1acd02bc554f35"),
229
+ /// ];
230
+ ///
231
+ /// unsorted.sort_unstable_by_key(PublicKey::to_sort_key);
232
+ ///
233
+ /// assert_eq!(unsorted, sorted);
234
+ /// ```
235
+ pub fn to_sort_key ( & self ) -> SortKey {
236
+ if self . compressed {
237
+ let bytes = self . inner . serialize ( ) ;
238
+ let mut res = [ 0 ; 32 ] ;
239
+ res[ ..] . copy_from_slice ( & bytes[ 1 ..33 ] ) ;
240
+ SortKey ( bytes[ 0 ] , res, [ 0 ; 32 ] )
241
+ } else {
242
+ let bytes = self . inner . serialize_uncompressed ( ) ;
243
+ let mut res_left = [ 0 ; 32 ] ;
244
+ let mut res_right = [ 0 ; 32 ] ;
245
+ res_left[ ..] . copy_from_slice ( & bytes[ 1 ..33 ] ) ;
246
+ res_right[ ..] . copy_from_slice ( & bytes[ 33 ..65 ] ) ;
247
+ SortKey ( bytes[ 0 ] , res_left, res_right)
248
+ }
249
+ }
250
+
185
251
/// Deserialize a public key from a slice
186
252
pub fn from_slice ( data : & [ u8 ] ) -> Result < PublicKey , Error > {
187
253
let compressed = match data. len ( ) {
@@ -208,6 +274,10 @@ impl PublicKey {
208
274
}
209
275
}
210
276
277
+ /// An opaque return type for PublicKey::to_sort_key
278
+ #[ derive( Debug , Hash , PartialEq , Eq , PartialOrd , Ord , Clone , Copy ) ]
279
+ pub struct SortKey ( u8 , [ u8 ; 32 ] , [ u8 ; 32 ] ) ;
280
+
211
281
impl fmt:: Display for PublicKey {
212
282
fn fmt ( & self , f : & mut fmt:: Formatter ) -> fmt:: Result {
213
283
if self . compressed {
@@ -485,10 +555,10 @@ impl<'de> ::serde::Deserialize<'de> for PublicKey {
485
555
#[ cfg( test) ]
486
556
mod tests {
487
557
use crate :: io;
488
- use super :: { PrivateKey , PublicKey } ;
558
+ use super :: { PrivateKey , PublicKey , SortKey } ;
489
559
use secp256k1:: Secp256k1 ;
490
560
use std:: str:: FromStr ;
491
- use crate :: hashes:: hex:: ToHex ;
561
+ use crate :: hashes:: hex:: { FromHex , ToHex } ;
492
562
use crate :: network:: constants:: Network :: Testnet ;
493
563
use crate :: network:: constants:: Network :: Bitcoin ;
494
564
use crate :: util:: address:: Address ;
@@ -639,4 +709,141 @@ mod tests {
639
709
assert ! ( PublicKey :: read_from( io:: Cursor :: new( & [ 0 ; 65 ] [ ..] ) ) . is_err( ) ) ;
640
710
assert ! ( PublicKey :: read_from( io:: Cursor :: new( & [ 4 ; 64 ] [ ..] ) ) . is_err( ) ) ;
641
711
}
712
+
713
+ #[ test]
714
+ fn pubkey_to_sort_key ( ) {
715
+ let key1 = PublicKey :: from_str ( "02ff12471208c14bd580709cb2358d98975247d8765f92bc25eab3b2763ed605f8" ) . unwrap ( ) ;
716
+ let key2 = PublicKey {
717
+ inner : key1. inner ,
718
+ compressed : false ,
719
+ } ;
720
+ let expected1 = SortKey (
721
+ 2 ,
722
+ <[ u8 ; 32 ] >:: from_hex (
723
+ "ff12471208c14bd580709cb2358d98975247d8765f92bc25eab3b2763ed605f8" ,
724
+ ) . unwrap ( ) ,
725
+ [ 0_u8 ; 32 ] ,
726
+ ) ;
727
+ let expected2 = SortKey (
728
+ 4 ,
729
+ <[ u8 ; 32 ] >:: from_hex (
730
+ "ff12471208c14bd580709cb2358d98975247d8765f92bc25eab3b2763ed605f8" ,
731
+ ) . unwrap ( ) ,
732
+ <[ u8 ; 32 ] >:: from_hex (
733
+ "1794e7f3d5e420641a3bc690067df5541470c966cbca8c694bf39aa16d836918" ,
734
+ ) . unwrap ( ) ,
735
+ ) ;
736
+ assert_eq ! ( key1. to_sort_key( ) , expected1) ;
737
+ assert_eq ! ( key2. to_sort_key( ) , expected2) ;
738
+ }
739
+
740
+ #[ test]
741
+ fn pubkey_sort ( ) {
742
+ struct Vector {
743
+ input : Vec < PublicKey > ,
744
+ expect : Vec < PublicKey > ,
745
+ }
746
+ let fmt = |v : Vec < _ > | v. into_iter ( )
747
+ . map ( |s| PublicKey :: from_str ( s) . unwrap ( ) )
748
+ . collect :: < Vec < _ > > ( ) ;
749
+ let vectors = vec ! [
750
+ // Start BIP67 vectors
751
+ // Vector 1
752
+ Vector {
753
+ input: fmt( vec![
754
+ "02ff12471208c14bd580709cb2358d98975247d8765f92bc25eab3b2763ed605f8" ,
755
+ "02fe6f0a5a297eb38c391581c4413e084773ea23954d93f7753db7dc0adc188b2f" ,
756
+ ] ) ,
757
+ expect: fmt( vec![
758
+ "02fe6f0a5a297eb38c391581c4413e084773ea23954d93f7753db7dc0adc188b2f" ,
759
+ "02ff12471208c14bd580709cb2358d98975247d8765f92bc25eab3b2763ed605f8" ,
760
+ ] ) ,
761
+ } ,
762
+ // Vector 2 (Already sorted, no action required)
763
+ Vector {
764
+ input: fmt( vec![
765
+ "02632b12f4ac5b1d1b72b2a3b508c19172de44f6f46bcee50ba33f3f9291e47ed0" ,
766
+ "027735a29bae7780a9755fae7a1c4374c656ac6a69ea9f3697fda61bb99a4f3e77" ,
767
+ "02e2cc6bd5f45edd43bebe7cb9b675f0ce9ed3efe613b177588290ad188d11b404" ,
768
+ ] ) ,
769
+ expect: fmt( vec![
770
+ "02632b12f4ac5b1d1b72b2a3b508c19172de44f6f46bcee50ba33f3f9291e47ed0" ,
771
+ "027735a29bae7780a9755fae7a1c4374c656ac6a69ea9f3697fda61bb99a4f3e77" ,
772
+ "02e2cc6bd5f45edd43bebe7cb9b675f0ce9ed3efe613b177588290ad188d11b404" ,
773
+ ] ) ,
774
+ } ,
775
+ // Vector 3
776
+ Vector {
777
+ input: fmt( vec![
778
+ "030000000000000000000000000000000000004141414141414141414141414141" ,
779
+ "020000000000000000000000000000000000004141414141414141414141414141" ,
780
+ "020000000000000000000000000000000000004141414141414141414141414140" ,
781
+ "030000000000000000000000000000000000004141414141414141414141414140" ,
782
+ ] ) ,
783
+ expect: fmt( vec![
784
+ "020000000000000000000000000000000000004141414141414141414141414140" ,
785
+ "020000000000000000000000000000000000004141414141414141414141414141" ,
786
+ "030000000000000000000000000000000000004141414141414141414141414140" ,
787
+ "030000000000000000000000000000000000004141414141414141414141414141" ,
788
+ ] ) ,
789
+ } ,
790
+ // Vector 4: (from bitcore)
791
+ Vector {
792
+ input: fmt( vec![
793
+ "022df8750480ad5b26950b25c7ba79d3e37d75f640f8e5d9bcd5b150a0f85014da" ,
794
+ "03e3818b65bcc73a7d64064106a859cc1a5a728c4345ff0b641209fba0d90de6e9" ,
795
+ "021f2f6e1e50cb6a953935c3601284925decd3fd21bc445712576873fb8c6ebc18" ,
796
+ ] ) ,
797
+ expect: fmt( vec![
798
+ "021f2f6e1e50cb6a953935c3601284925decd3fd21bc445712576873fb8c6ebc18" ,
799
+ "022df8750480ad5b26950b25c7ba79d3e37d75f640f8e5d9bcd5b150a0f85014da" ,
800
+ "03e3818b65bcc73a7d64064106a859cc1a5a728c4345ff0b641209fba0d90de6e9" ,
801
+ ] ) ,
802
+ } ,
803
+ // Non-BIP67 vectors
804
+ Vector {
805
+ input: fmt( vec![
806
+ "02c690d642c1310f3a1ababad94e3930e4023c930ea472e7f37f660fe485263b88" ,
807
+ "0234dd69c56c36a41230d573d68adeae0030c9bc0bf26f24d3e1b64c604d293c68" ,
808
+ "041a181bd0e79974bd7ca552e09fc42ba9c3d5dbb3753741d6f0ab3015dbfd9a22d6b001a32f5f51ac6f2c0f35e73a6a62f59e848fa854d3d21f3f231594eeaa46" ,
809
+ "032b8324c93575034047a52e9bca05a46d8347046b91a032eff07d5de8d3f2730b" ,
810
+ "04c4b0bbb339aa236bff38dbe6a451e111972a7909a126bc424013cba2ec33bc3816753d96001fd7cba3ce5372f5c9a0d63708183033538d07b1e532fc43aaacfa" ,
811
+ "028e1c947c8c0b8ed021088b8e981491ac7af2b8fabebea1abdb448424c8ed75b7" ,
812
+ "045d753414fa292ea5b8f56e39cfb6a0287b2546231a5cb05c4b14ab4b463d171f5128148985b23eccb1e2905374873b1f09b9487f47afa6b1f2b0083ac8b4f7e8" ,
813
+ "03004a8a3d242d7957c0b60fb7208d386fa6a0193aabd1f3f095ffd0ac097e447b" ,
814
+ "04eb0db2d71ccbb0edd8fb35092cbcae2f7fa1f06d4c170804bf52007924b569a8d2d6f6bc8fd2b3caa3253fa1bb674443743bf7fb9f94f9c0b0831a252894cfa8" ,
815
+ "04516cde23e14f2319423b7a4a7ae48b1dadceb5e9c123198d417d10895684c42eb05e210f90ccbc72448803a22312e3f122ff2939956ccef4f7316f836295ddd5" ,
816
+ "038f47dcd43ba6d97fc9ed2e3bba09b175a45fac55f0683e8cf771e8ced4572354" ,
817
+ "04c6bec3b07586a4b085a78cbb97e9bab6f1d3c9ebf299b65dec85213c5eacd44487de86017183120bb7ea3b6c6660c5037615fe1add2a73f800cbeeae22c60438" ,
818
+ "03e1a1cfa9eaff604ae237b7af31ffe4c01be22eb96f3da0e62c5850dd4b4386c1" ,
819
+ "028d3a2d9f1b1c5c75845944f93bc183ba23aecde53f1978b8aa1b77661be6114f" ,
820
+ "028bde91b10013e08949a318018fedbd896534a549a278e220169ee2a36517c7aa" ,
821
+ "04c4b0bbb339aa236bff38dbe6a451e111972a7909a126bc424013cba2ec33bc38e98ac269ffe028345c31ac8d0a365f29c8f7e7cfccac72f84e1acd02bc554f35" ,
822
+ ] ) ,
823
+ expect: fmt( vec![
824
+ "0234dd69c56c36a41230d573d68adeae0030c9bc0bf26f24d3e1b64c604d293c68" ,
825
+ "028bde91b10013e08949a318018fedbd896534a549a278e220169ee2a36517c7aa" ,
826
+ "028d3a2d9f1b1c5c75845944f93bc183ba23aecde53f1978b8aa1b77661be6114f" ,
827
+ "028e1c947c8c0b8ed021088b8e981491ac7af2b8fabebea1abdb448424c8ed75b7" ,
828
+ "02c690d642c1310f3a1ababad94e3930e4023c930ea472e7f37f660fe485263b88" ,
829
+ "03004a8a3d242d7957c0b60fb7208d386fa6a0193aabd1f3f095ffd0ac097e447b" ,
830
+ "032b8324c93575034047a52e9bca05a46d8347046b91a032eff07d5de8d3f2730b" ,
831
+ "038f47dcd43ba6d97fc9ed2e3bba09b175a45fac55f0683e8cf771e8ced4572354" ,
832
+ "03e1a1cfa9eaff604ae237b7af31ffe4c01be22eb96f3da0e62c5850dd4b4386c1" ,
833
+ "041a181bd0e79974bd7ca552e09fc42ba9c3d5dbb3753741d6f0ab3015dbfd9a22d6b001a32f5f51ac6f2c0f35e73a6a62f59e848fa854d3d21f3f231594eeaa46" ,
834
+ "04516cde23e14f2319423b7a4a7ae48b1dadceb5e9c123198d417d10895684c42eb05e210f90ccbc72448803a22312e3f122ff2939956ccef4f7316f836295ddd5" ,
835
+ "045d753414fa292ea5b8f56e39cfb6a0287b2546231a5cb05c4b14ab4b463d171f5128148985b23eccb1e2905374873b1f09b9487f47afa6b1f2b0083ac8b4f7e8" ,
836
+ // These two pubkeys are mirrored. This helps verify the sort past the x value.
837
+ "04c4b0bbb339aa236bff38dbe6a451e111972a7909a126bc424013cba2ec33bc3816753d96001fd7cba3ce5372f5c9a0d63708183033538d07b1e532fc43aaacfa" ,
838
+ "04c4b0bbb339aa236bff38dbe6a451e111972a7909a126bc424013cba2ec33bc38e98ac269ffe028345c31ac8d0a365f29c8f7e7cfccac72f84e1acd02bc554f35" ,
839
+ "04c6bec3b07586a4b085a78cbb97e9bab6f1d3c9ebf299b65dec85213c5eacd44487de86017183120bb7ea3b6c6660c5037615fe1add2a73f800cbeeae22c60438" ,
840
+ "04eb0db2d71ccbb0edd8fb35092cbcae2f7fa1f06d4c170804bf52007924b569a8d2d6f6bc8fd2b3caa3253fa1bb674443743bf7fb9f94f9c0b0831a252894cfa8" ,
841
+ ] ) ,
842
+ } ,
843
+ ] ;
844
+ for mut vector in vectors {
845
+ vector. input . sort_by_cached_key ( PublicKey :: to_sort_key) ;
846
+ assert_eq ! ( vector. input, vector. expect) ;
847
+ }
848
+ }
642
849
}
0 commit comments