-
Notifications
You must be signed in to change notification settings - Fork 20
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Currently, we only support using our internal hash type, that is the 32-bytes representation of a sha512-256 hash digest. However, we may want to give callers the possibility to use other hash types, or store them in other ways. This commit modifies `Pollard`, `Proof` and `Stump` to take in a generic parameter `Hash` that implements the trait `NodeHash` and defaults to `BitcoinNodeHash`, the one used by Bitcoin consensus as defined by the utreexo spec. This is part of a project to support a Cairo prover for Bitcoin. To reduce the circuit size, we need to use an algebraic hash function like Poseidon, which tends to reduce the circuit significantly. With this commit the caller can use our data structures with Poseidon without needing to change anything in rustreexo.
- Loading branch information
1 parent
6a8fe53
commit 508339e
Showing
10 changed files
with
915 additions
and
400 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,143 @@ | ||
//! All data structures in this library are generic over the hash type used, defaulting to | ||
//! [BitcoinNodeHash](crate::accumulator::node_hash::BitcoinNodeHash), the one used by Bitcoin | ||
//! as defined by the utreexo spec. However, if you need to use a different hash type, you can | ||
//! implement the [NodeHash](crate::accumulator::node_hash::NodeHash) trait for it, and use it | ||
//! with the accumulator data structures. | ||
//! | ||
//! This example shows how to use a custom hash type based on the Poseidon hash function. The | ||
//! [Poseidon Hash](https://eprint.iacr.org/2019/458.pdf) is a hash function that is optmized | ||
//! for zero-knowledge proofs, and is used in projects like ZCash and StarkNet. | ||
//! If you want to work with utreexo proofs in zero-knowledge you may want to use this instead | ||
//! of our usual sha512-256 that we use by default, since that will give you smaller circuits. | ||
//! This example shows how to use both the [Pollard](crate::accumulator::pollard::Pollard) and | ||
//! proofs with a custom hash type. The code here should be pretty much all you need to do to | ||
//! use your custom hashes, just tweak the implementation of | ||
//! [NodeHash](crate::accumulator::node_hash::NodeHash) for your hash type. | ||
use rustreexo::accumulator::node_hash::NodeHash; | ||
use rustreexo::accumulator::pollard::Pollard; | ||
use starknet_crypto::poseidon_hash_many; | ||
use starknet_crypto::Felt; | ||
|
||
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)] | ||
/// We need a stateful wrapper around the actual hash, this is because we use those different | ||
/// values inside our accumulator. Here we use an enum to represent the different states, you | ||
/// may want to use a struct with more data, depending on your needs. | ||
enum PoseidonHash { | ||
/// This means this holds an actual value | ||
/// | ||
/// It usually represents a node in the accumulator that haven't been deleted. | ||
Hash(Felt), | ||
/// Placeholder is a value that haven't been deleted, but we don't have the actual value. | ||
/// The only thing that matters about it is that it's not empty. You can implement this | ||
/// the way you want, just make sure that [NodeHash::is_placeholder] and [NodeHash::placeholder] | ||
/// returns sane values (that is, if we call [NodeHash::placeholder] calling [NodeHash::is_placeholder] | ||
/// on the result should return true). | ||
Placeholder, | ||
/// This is an empty value, it represents a node that was deleted from the accumulator. | ||
/// | ||
/// Same as the placeholder, you can implement this the way you want, just make sure that | ||
/// [NodeHash::is_empty] and [NodeHash::empty] returns sane values. | ||
Empty, | ||
} | ||
|
||
// you'll need to implement Display for your hash type, so you can print it. | ||
impl std::fmt::Display for PoseidonHash { | ||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||
match self { | ||
PoseidonHash::Hash(h) => write!(f, "Hash({})", h), | ||
PoseidonHash::Placeholder => write!(f, "Placeholder"), | ||
PoseidonHash::Empty => write!(f, "Empty"), | ||
} | ||
} | ||
} | ||
|
||
// this is the implementation of the NodeHash trait for our custom hash type. And it's the only | ||
// thing you need to do to use your custom hash type with the accumulator data structures. | ||
impl NodeHash for PoseidonHash { | ||
// returns a new placeholder type such that is_placeholder returns true | ||
fn placeholder() -> Self { | ||
PoseidonHash::Placeholder | ||
} | ||
|
||
// returns an empty hash such that is_empty returns true | ||
fn empty() -> Self { | ||
PoseidonHash::Empty | ||
} | ||
|
||
// returns true if this is a placeholder. This should be true iff this type was created by | ||
// calling placeholder. | ||
fn is_placeholder(&self) -> bool { | ||
matches!(self, PoseidonHash::Placeholder) | ||
} | ||
|
||
// returns true if this is an empty hash. This should be true iff this type was created by | ||
// calling empty. | ||
fn is_empty(&self) -> bool { | ||
matches!(self, PoseidonHash::Empty) | ||
} | ||
|
||
// used for serialization, writes the hash to the writer | ||
// | ||
// if you don't want to use serialization, you can just return an error here. | ||
fn write<W>(&self, writer: &mut W) -> std::io::Result<()> | ||
where | ||
W: std::io::Write, | ||
{ | ||
match self { | ||
PoseidonHash::Hash(h) => writer.write_all(&h.to_bytes_be()), | ||
PoseidonHash::Placeholder => writer.write_all(&[0u8; 32]), | ||
PoseidonHash::Empty => writer.write_all(&[0u8; 32]), | ||
} | ||
} | ||
|
||
// used for deserialization, reads the hash from the reader | ||
// | ||
// if you don't want to use serialization, you can just return an error here. | ||
fn read<R>(reader: &mut R) -> std::io::Result<Self> | ||
where | ||
R: std::io::Read, | ||
{ | ||
let mut buf = [0u8; 32]; | ||
reader.read_exact(&mut buf)?; | ||
if buf.iter().all(|&b| b == 0) { | ||
Ok(PoseidonHash::Empty) | ||
} else { | ||
Ok(PoseidonHash::Hash(Felt::from_bytes_be(&buf))) | ||
} | ||
} | ||
|
||
// the main thing about the hash type, it returns the next node's hash, given it's children. | ||
// The implementation of this method is highly consensus critical, so everywhere should use the | ||
// exact same algorithm to calculate the next hash. Rustreexo won't call this method, unless | ||
// **both** children are not empty. | ||
fn parent_hash(left: &Self, right: &Self) -> Self { | ||
if let (PoseidonHash::Hash(left), PoseidonHash::Hash(right)) = (left, right) { | ||
return PoseidonHash::Hash(poseidon_hash_many(&[*left, *right])); | ||
} | ||
|
||
// This should never happen, since rustreexo won't call this method unless both children | ||
// are not empty. | ||
unreachable!() | ||
} | ||
} | ||
|
||
fn main() { | ||
// Create a vector with two utxos that will be added to the Pollard | ||
let elements = vec![ | ||
PoseidonHash::Hash(Felt::from(1)), | ||
PoseidonHash::Hash(Felt::from(2)), | ||
]; | ||
|
||
// Create a new Pollard, and add the utxos to it | ||
let mut p = Pollard::<PoseidonHash>::new_with_hash(); | ||
p.modify(&elements, &[]).unwrap(); | ||
|
||
// Create a proof that the first utxo is in the Pollard | ||
let proof = p.prove(&[elements[0]]).unwrap(); | ||
|
||
// check that the proof has exactly one target | ||
assert_eq!(proof.n_targets(), 1); | ||
// check that the proof is what we expect | ||
assert!(p.verify(&proof, &[elements[0]]).unwrap()); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.