penumbra_sdk_tct/internal/hash/
option.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
use ark_ff::{BigInt, PrimeField};
use decaf377::Fq;
use std::fmt::Debug;

use crate::prelude::*;

/// A representation of `Option<Hash>` without the tag bytes required by `Option`, because we
/// know that no valid [`struct@Hash`] will be equal to `[u64::MAX; 4]`, since the modulus for
/// [`Commitment`](crate::Commitment) is too small.
///
/// This type is inter-convertible via [`From`] and [`Into`] with `Option<Hash>`, and that is
/// its only purpose.
#[derive(Clone, Copy, PartialEq, Eq, Serialize, Deserialize)]
#[serde(from = "Option<Hash>", into = "Option<Hash>")]
pub struct OptionHash {
    inner: [u64; 4],
}

impl Debug for OptionHash {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        <Option<Hash>>::from(*self).fmt(f)
    }
}

impl Default for OptionHash {
    fn default() -> Self {
        Self {
            inner: [u64::MAX; 4],
        }
    }
}

impl From<Option<Hash>> for OptionHash {
    fn from(hash: Option<Hash>) -> Self {
        match hash {
            Some(hash) => Self {
                inner: Fq::into_bigint(hash.0).0,
            },
            None => Self {
                // This sentinel value is not a valid `Fq` because it's bigger than the modulus,
                // which means that it will never occur otherwise
                inner: [u64::MAX; 4],
            },
        }
    }
}

impl From<OptionHash> for Option<Hash> {
    fn from(hash: OptionHash) -> Self {
        if hash.inner == [u64::MAX; 4] {
            None
        } else {
            // We're directly constructing the hash here by coercing the bytes into the right type,
            // but this is safe because we know that the bytes are a real `Fq` and not the sentinel
            // value we just checked for
            Some(Hash::new(
                Fq::from_bigint(BigInt::new(hash.inner))
                    .expect("convert bytes to a valid Fq element"),
            ))
        }
    }
}