// -*- coding: utf-8; mode: rust; -*- // // To the extent possible under law, the authors have waived all // copyright and related or neighboring rights to zkp, // using the Creative Commons "CC0" public domain dedication. See // for full // details. // // Authors: // - Henry de Valence extern crate rand; use rand::{thread_rng, CryptoRng, RngCore}; extern crate curve25519_dalek; use curve25519_dalek::constants as dalek_constants; use curve25519_dalek::ristretto::{CompressedRistretto, RistrettoPoint}; use curve25519_dalek::scalar::Scalar; #[macro_use] extern crate lox_zkp; pub use lox_zkp::Transcript; define_proof! {sig_proof, "Sig", (x), (A), (B) : A = (x * B) } define_proof! {vrf_proof, "VRF", (x), (A, G, H), (B) : A = (x * B), G = (x * H) } /// Defines how the construction interacts with the transcript. trait TranscriptProtocol { fn append_message_example(&mut self, message: &[u8]); fn hash_to_group(self) -> RistrettoPoint; } impl TranscriptProtocol for Transcript { fn append_message_example(&mut self, message: &[u8]) { self.append_message(b"msg", message); } fn hash_to_group(mut self) -> RistrettoPoint { let mut bytes = [0u8; 64]; self.challenge_bytes(b"output", &mut bytes); RistrettoPoint::from_uniform_bytes(&bytes) } } #[derive(Clone)] pub struct SecretKey(Scalar); impl SecretKey { fn new(rng: &mut R) -> SecretKey { SecretKey(Scalar::random(rng)) } } #[derive(Copy, Clone)] pub struct PublicKey(RistrettoPoint, CompressedRistretto); impl<'a> From<&'a SecretKey> for PublicKey { fn from(sk: &'a SecretKey) -> PublicKey { let pk = &sk.0 * dalek_constants::RISTRETTO_BASEPOINT_TABLE; PublicKey(pk, pk.compress()) } } pub struct KeyPair { sk: SecretKey, pk: PublicKey, } impl From for KeyPair { fn from(sk: SecretKey) -> KeyPair { let pk = PublicKey::from(&sk); KeyPair { sk, pk } } } pub struct Signature(sig_proof::BatchableProof); pub struct VrfOutput(CompressedRistretto); pub struct VrfProof(vrf_proof::CompactProof); impl KeyPair { fn public_key(&self) -> PublicKey { self.pk } fn sign(&self, message: &[u8], sig_transcript: &mut Transcript) -> Signature { sig_transcript.append_message_example(message); let (proof, _points) = sig_proof::prove_batchable( sig_transcript, sig_proof::ProveAssignments { x: &self.sk.0, A: &self.pk.0, B: &dalek_constants::RISTRETTO_BASEPOINT_POINT, }, ); Signature(proof) } #[allow(non_snake_case)] fn vrf( &self, mut function_transcript: Transcript, message: &[u8], proof_transcript: &mut Transcript, ) -> (VrfOutput, VrfProof) { // Use function_transcript to hash the message to a point H function_transcript.append_message_example(message); let H = function_transcript.hash_to_group(); // Compute the VRF output G and form a proof let G = &H * &self.sk.0; let (proof, points) = vrf_proof::prove_compact( proof_transcript, vrf_proof::ProveAssignments { x: &self.sk.0, A: &self.pk.0, B: &dalek_constants::RISTRETTO_BASEPOINT_POINT, G: &G, H: &H, }, ); (VrfOutput(points.G), VrfProof(proof)) } } impl Signature { fn verify( &self, message: &[u8], pubkey: &PublicKey, sig_transcript: &mut Transcript, ) -> Result<(), ()> { sig_transcript.append_message_example(message); sig_proof::verify_batchable( &self.0, sig_transcript, sig_proof::VerifyAssignments { A: &pubkey.1, B: &dalek_constants::RISTRETTO_BASEPOINT_COMPRESSED, }, ) .map_err(|_discard_error_info_in_test_code| ()) } } impl VrfOutput { #[allow(non_snake_case)] fn verify( &self, mut function_transcript: Transcript, message: &[u8], pubkey: &PublicKey, proof_transcript: &mut Transcript, proof: &VrfProof, ) -> Result<(), ()> { // Use function_transcript to hash the message to a point H function_transcript.append_message_example(message); let H = function_transcript.hash_to_group().compress(); vrf_proof::verify_compact( &proof.0, proof_transcript, vrf_proof::VerifyAssignments { A: &pubkey.1, B: &dalek_constants::RISTRETTO_BASEPOINT_COMPRESSED, G: &self.0, H: &H, }, ) .map_err(|_discard_error_info_in_test_code| ()) } } #[test] fn create_and_verify_sig() { let domain_sep = b"My Sig Application"; let msg1 = b"Test Message 1"; let msg2 = b"Test Message 2"; let kp1 = KeyPair::from(SecretKey::new(&mut thread_rng())); let pk1 = kp1.public_key(); let kp2 = KeyPair::from(SecretKey::new(&mut thread_rng())); let pk2 = kp2.public_key(); let sig1 = kp1.sign(&msg1[..], &mut Transcript::new(domain_sep)); let sig2 = kp2.sign(&msg2[..], &mut Transcript::new(domain_sep)); // Check that each signature verifies assert!(sig1 .verify(msg1, &pk1, &mut Transcript::new(domain_sep),) .is_ok()); assert!(sig2 .verify(msg2, &pk2, &mut Transcript::new(domain_sep),) .is_ok()); // Check that verification with the wrong pubkey fails assert!(sig1 .verify(msg1, &pk2, &mut Transcript::new(domain_sep),) .is_err()); assert!(sig2 .verify(msg2, &pk1, &mut Transcript::new(domain_sep),) .is_err()); // Check that verification with the wrong message fails assert!(sig1 .verify(msg2, &pk1, &mut Transcript::new(domain_sep),) .is_err()); assert!(sig2 .verify(msg1, &pk2, &mut Transcript::new(domain_sep),) .is_err()); // Check that verification with the wrong domain separator fails assert!(sig1 .verify(msg1, &pk1, &mut Transcript::new(b"Wrong"),) .is_err()); assert!(sig2 .verify(msg2, &pk2, &mut Transcript::new(b"Wrong"),) .is_err()); } #[test] #[ignore] fn create_and_verify_bigsig() { let domain_sep = b"My Sig Application"; let mut large_msg = Vec::new(); large_msg.resize((u32::max_value() as usize) + 250, 1u8); let kp = KeyPair::from(SecretKey::new(&mut thread_rng())); let pk = kp.public_key(); let sig = kp.sign(&large_msg[..], &mut Transcript::new(domain_sep)); // Check that the signature verifies (& doesn't panic inside Merlin) assert!(sig .verify(&large_msg[..], &pk, &mut Transcript::new(domain_sep),) .is_ok()); } #[test] fn counterparty_signature_chain() { let domain_sep = b"Counterparty Example"; let msg1a = b"In this test, two counterparties exchange signatures."; let msg2a = b"However, the counterparties sign and verify messages"; let msg1b = b"using stateful transcript objects."; let msg2b = b"When party 1 signs, the party 1 transcript changes;"; let msg1c = b"when party 2 verifies, the party 2 transcript syncs."; let msg2c = b"In this way, the transcript states ratchet stateful signatures."; let kp1 = KeyPair::from(SecretKey::new(&mut thread_rng())); let pk1 = kp1.public_key(); let kp2 = KeyPair::from(SecretKey::new(&mut thread_rng())); let pk2 = kp2.public_key(); let mut trans1 = Transcript::new(domain_sep); let mut trans2 = Transcript::new(domain_sep); // Round a, Party 1 -----> Party 2 let sig1a = kp1.sign(&msg1a[..], &mut trans1); assert!(sig1a.verify(msg1a, &pk1, &mut trans2).is_ok()); // Round a, Party 2 -----> Party 1 let sig2a = kp2.sign(&msg2a[..], &mut trans2); assert!(sig2a.verify(msg2a, &pk2, &mut trans1).is_ok()); // Round b, Party 1 -----> Party 2 let sig1b = kp1.sign(&msg1b[..], &mut trans1); assert!(sig1b.verify(msg1b, &pk1, &mut trans2).is_ok()); // Round b, Party 2 -----> Party 1 let sig2b = kp2.sign(&msg2b[..], &mut trans2); assert!(sig2b.verify(msg2b, &pk2, &mut trans1).is_ok()); // Round c, Party 1 -----> Party 2 let sig1c = kp1.sign(&msg1c[..], &mut trans1); assert!(sig1c.verify(msg1c, &pk1, &mut trans2).is_ok()); // Round c, Party 2 -----> Party 1 let sig2c = kp2.sign(&msg2c[..], &mut trans2); assert!(sig2c.verify(msg2c, &pk2, &mut trans1).is_ok()); } #[test] fn create_and_verify_vrf() { let domain_sep = b"My VRF Application"; let msg1 = b"Test Message 1"; let msg2 = b"Test Message 2"; let kp1 = KeyPair::from(SecretKey::new(&mut thread_rng())); let pk1 = kp1.public_key(); let kp2 = KeyPair::from(SecretKey::new(&mut thread_rng())); let pk2 = kp2.public_key(); let (output1, proof1) = kp1.vrf( Transcript::new(domain_sep), &msg1[..], &mut Transcript::new(domain_sep), ); let (output2, proof2) = kp2.vrf( Transcript::new(domain_sep), &msg2[..], &mut Transcript::new(domain_sep), ); // Check that each VRF output was correctly produced assert!(output1 .verify( Transcript::new(domain_sep), msg1, &pk1, &mut Transcript::new(domain_sep), &proof1, ) .is_ok()); assert!(output2 .verify( Transcript::new(domain_sep), msg2, &pk2, &mut Transcript::new(domain_sep), &proof2, ) .is_ok()); // Check that verification with the wrong pubkey fails assert!(output1 .verify( Transcript::new(domain_sep), msg1, &pk2, // swap pubkey &mut Transcript::new(domain_sep), &proof1, ) .is_err()); assert!(output2 .verify( Transcript::new(domain_sep), msg2, &pk1, // swap pubkey &mut Transcript::new(domain_sep), &proof2, ) .is_err()); // Check that verification with the wrong output fails assert!(output2 // swap output .verify( Transcript::new(domain_sep), msg1, &pk1, &mut Transcript::new(domain_sep), &proof1, ) .is_err()); assert!(output1 // swap output .verify( Transcript::new(domain_sep), msg2, &pk2, &mut Transcript::new(domain_sep), &proof2, ) .is_err()); // Check that verification with the wrong domain separator fails assert!(output1 .verify( Transcript::new(domain_sep), msg1, &pk1, &mut Transcript::new(b"A different application"), // swap dom-sep &proof1, ) .is_err()); assert!(output2 .verify( Transcript::new(domain_sep), msg2, &pk2, &mut Transcript::new(b"A different application"), // swap dom-sep &proof2, ) .is_err()); }