fedimint_core/encoding/
secp256k1.rsuse std::io::{Error, Read, Write};
use secp256k1_zkp::ecdsa::Signature;
use crate::encoding::{Decodable, DecodeError, Encodable};
use crate::module::registry::ModuleDecoderRegistry;
impl Encodable for secp256k1_zkp::ecdsa::Signature {
fn consensus_encode<W: std::io::Write>(&self, writer: &mut W) -> Result<usize, std::io::Error> {
let bytes = self.serialize_compact();
writer.write_all(&bytes)?;
Ok(bytes.len())
}
}
impl Decodable for secp256k1_zkp::ecdsa::Signature {
fn consensus_decode<D: std::io::Read>(
d: &mut D,
modules: &ModuleDecoderRegistry,
) -> Result<Self, DecodeError> {
Signature::from_compact(&<[u8; 64]>::consensus_decode(d, modules)?)
.map_err(DecodeError::from_err)
}
}
impl Encodable for secp256k1_zkp::PublicKey {
fn consensus_encode<W: std::io::Write>(&self, writer: &mut W) -> Result<usize, std::io::Error> {
self.serialize().consensus_encode(writer)
}
}
impl Decodable for secp256k1_zkp::PublicKey {
fn consensus_decode<D: std::io::Read>(
d: &mut D,
modules: &ModuleDecoderRegistry,
) -> Result<Self, DecodeError> {
secp256k1_zkp::PublicKey::from_slice(&<[u8; 33]>::consensus_decode(d, modules)?)
.map_err(DecodeError::from_err)
}
}
impl Encodable for secp256k1_zkp::SecretKey {
fn consensus_encode<W: std::io::Write>(&self, writer: &mut W) -> Result<usize, std::io::Error> {
self.secret_bytes().consensus_encode(writer)
}
}
impl Decodable for secp256k1_zkp::SecretKey {
fn consensus_decode<D: std::io::Read>(
d: &mut D,
modules: &ModuleDecoderRegistry,
) -> Result<Self, DecodeError> {
secp256k1_zkp::SecretKey::from_slice(&<[u8; 32]>::consensus_decode(d, modules)?)
.map_err(DecodeError::from_err)
}
}
impl Encodable for secp256k1_zkp::schnorr::Signature {
fn consensus_encode<W: std::io::Write>(&self, writer: &mut W) -> Result<usize, std::io::Error> {
let bytes = &self[..];
assert_eq!(
bytes.len(),
secp256k1_zkp::constants::SCHNORR_SIGNATURE_SIZE
);
writer.write_all(bytes)?;
Ok(secp256k1_zkp::constants::SCHNORR_SIGNATURE_SIZE)
}
}
impl Decodable for secp256k1_zkp::schnorr::Signature {
fn consensus_decode<D: std::io::Read>(
d: &mut D,
modules: &ModuleDecoderRegistry,
) -> Result<Self, DecodeError> {
let bytes =
<[u8; secp256k1_zkp::constants::SCHNORR_SIGNATURE_SIZE]>::consensus_decode(d, modules)?;
secp256k1_zkp::schnorr::Signature::from_slice(&bytes).map_err(DecodeError::from_err)
}
}
impl Encodable for bitcoin::key::KeyPair {
fn consensus_encode<W: Write>(&self, writer: &mut W) -> Result<usize, Error> {
self.secret_bytes().consensus_encode(writer)
}
}
impl Decodable for bitcoin::key::KeyPair {
fn consensus_decode<D: Read>(
d: &mut D,
modules: &ModuleDecoderRegistry,
) -> Result<Self, DecodeError> {
let sec_bytes = <[u8; 32]>::consensus_decode(d, modules)?;
Self::from_seckey_slice(secp256k1_zkp::global::SECP256K1, &sec_bytes) .map_err(DecodeError::from_err)
}
}
#[cfg(test)]
mod tests {
use secp256k1_zkp::hashes::Hash as BitcoinHash;
use secp256k1_zkp::Message;
use super::super::tests::test_roundtrip;
#[test_log::test]
fn test_ecdsa_sig() {
let ctx = secp256k1_zkp::Secp256k1::new();
let (sk, _pk) = ctx.generate_keypair(&mut rand::thread_rng());
let sig = ctx.sign_ecdsa(
&Message::from_hashed_data::<secp256k1_zkp::hashes::sha256::Hash>(b"Hello World!"),
&sk,
);
test_roundtrip(&sig);
}
#[test_log::test]
fn test_schnorr_pub_key() {
let ctx = secp256k1_zkp::global::SECP256K1;
let mut rng = rand::rngs::OsRng;
let sec_key = bitcoin::key::KeyPair::new(ctx, &mut rng);
let pub_key = sec_key.public_key();
test_roundtrip(&pub_key);
let sig = ctx.sign_schnorr(
&secp256k1_zkp::hashes::sha256::Hash::hash(b"Hello World!").into(),
&sec_key,
);
test_roundtrip(&sig);
}
}