|
1 | 1 | use std::hash::Hasher as StdHasher; |
2 | 2 |
|
3 | 3 | use bitvec::{self, BitVec}; |
| 4 | +use byteorder::{LittleEndian, ReadBytesExt, WriteBytesExt}; |
4 | 5 | use merkle_light::hash::{Algorithm as LightAlgorithm, Hashable}; |
5 | 6 | use pairing::bls12_381::{Bls12, Fr, FrRepr}; |
6 | 7 | use pairing::{PrimeField, PrimeFieldRepr}; |
7 | 8 | use rand::{Rand, Rng}; |
8 | 9 | use sapling_crypto::pedersen_hash::{pedersen_hash, Personalization}; |
| 10 | +use serde::de::{Deserialize, Deserializer}; |
| 11 | +use serde::ser::Serializer; |
9 | 12 |
|
10 | 13 | use super::{Domain, HashFunction, Hasher}; |
11 | 14 | use crate::crypto::{kdf, pedersen, sloth}; |
@@ -61,8 +64,52 @@ impl Hashable<PedersenFunction> for PedersenDomain { |
61 | 64 | } |
62 | 65 | } |
63 | 66 |
|
64 | | -#[derive(Copy, Clone, PartialEq, Eq, Debug)] |
65 | | -pub struct PedersenDomain(pub FrRepr); |
| 67 | +#[derive(Copy, Clone, PartialEq, Eq, Debug, Serialize, Deserialize)] |
| 68 | +pub struct PedersenDomain(#[serde(with = "FrReprDef")] pub FrRepr); |
| 69 | + |
| 70 | +pub struct FrReprDef(pub [u64; 4]); |
| 71 | + |
| 72 | +impl FrReprDef { |
| 73 | + fn serialize<S>(__self: &FrRepr, serializer: S) -> ::std::result::Result<S::Ok, S::Error> |
| 74 | + where |
| 75 | + S: Serializer, |
| 76 | + { |
| 77 | + let mut writer = Vec::with_capacity(32); |
| 78 | + |
| 79 | + for digit in __self.0.as_ref().iter() { |
| 80 | + writer.write_u64::<LittleEndian>(*digit).unwrap(); |
| 81 | + } |
| 82 | + |
| 83 | + if serializer.is_human_readable() { |
| 84 | + serializer.collect_str(&base64::display::Base64Display::with_config( |
| 85 | + &writer, |
| 86 | + base64::STANDARD, |
| 87 | + )) |
| 88 | + } else { |
| 89 | + serializer.serialize_bytes(&writer) |
| 90 | + } |
| 91 | + } |
| 92 | + |
| 93 | + fn deserialize<'de, D>(deserializer: D) -> ::std::result::Result<FrRepr, D::Error> |
| 94 | + where |
| 95 | + D: Deserializer<'de>, |
| 96 | + { |
| 97 | + let arr: Vec<u8> = if deserializer.is_human_readable() { |
| 98 | + let raw = String::deserialize(deserializer)?; |
| 99 | + base64::decode(&raw).unwrap() |
| 100 | + } else { |
| 101 | + Vec::deserialize(deserializer)? |
| 102 | + }; |
| 103 | + |
| 104 | + let mut digits = [0u64; 4]; |
| 105 | + let mut source = ::std::io::Cursor::new(arr); |
| 106 | + for digit in digits.iter_mut() { |
| 107 | + *digit = source.read_u64::<LittleEndian>().unwrap(); |
| 108 | + } |
| 109 | + |
| 110 | + Ok(FrRepr(digits)) |
| 111 | + } |
| 112 | +} |
66 | 113 |
|
67 | 114 | impl Default for PedersenDomain { |
68 | 115 | fn default() -> PedersenDomain { |
@@ -324,4 +371,15 @@ mod tests { |
324 | 371 | } |
325 | 372 | } |
326 | 373 | } |
| 374 | + |
| 375 | + #[test] |
| 376 | + fn test_serialize() { |
| 377 | + let repr = FrRepr([1, 2, 3, 4]); |
| 378 | + let val = PedersenDomain(repr); |
| 379 | + |
| 380 | + let ser = serde_json::to_string(&val).unwrap(); |
| 381 | + let val_back = serde_json::from_str(&ser).unwrap(); |
| 382 | + |
| 383 | + assert_eq!(val, val_back); |
| 384 | + } |
327 | 385 | } |
0 commit comments