use core::fmt;
use k256::ecdsa::{RecoveryId, Signature};
use crate::error::SignatureError;
pub const SIGNATURE_LEN: usize = 65;
#[derive(Clone, Copy, PartialEq, Eq, Hash)]
pub struct RecoverableSignature {
r: [u8; 32],
s: [u8; 32],
v: u8,
}
impl RecoverableSignature {
pub fn from_signature(sig: &Signature, recovery_id: RecoveryId) -> Self {
let bytes = sig.to_bytes();
let mut r = [0u8; 32];
let mut s = [0u8; 32];
r.copy_from_slice(&bytes[..32]);
s.copy_from_slice(&bytes[32..]);
Self {
r,
s,
v: recovery_id.to_byte(),
}
}
pub fn from_bytes(bytes: &[u8]) -> Result<Self, SignatureError> {
if bytes.len() != SIGNATURE_LEN {
return Err(SignatureError::BadLength(bytes.len()));
}
let v = match bytes[64] {
v @ (0 | 1) => v,
27 => 0,
28 => 1,
other => return Err(SignatureError::BadRecoveryId(other)),
};
let mut r = [0u8; 32];
let mut s = [0u8; 32];
r.copy_from_slice(&bytes[..32]);
s.copy_from_slice(&bytes[32..64]);
Ok(Self { r, s, v })
}
pub fn r(&self) -> &[u8; 32] {
&self.r
}
pub fn s(&self) -> &[u8; 32] {
&self.s
}
pub fn v(&self) -> u8 {
self.v
}
pub fn to_bytes(&self) -> [u8; SIGNATURE_LEN] {
let mut out = [0u8; SIGNATURE_LEN];
out[..32].copy_from_slice(&self.r);
out[32..64].copy_from_slice(&self.s);
out[64] = self.v;
out
}
pub fn split(&self) -> Result<(Signature, RecoveryId), SignatureError> {
let mut rs = [0u8; 64];
rs[..32].copy_from_slice(&self.r);
rs[32..].copy_from_slice(&self.s);
let sig = Signature::from_slice(&rs)?;
let recid =
RecoveryId::from_byte(self.v).ok_or(SignatureError::BadRecoveryId(self.v))?;
Ok((sig, recid))
}
}
impl fmt::Debug for RecoverableSignature {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "RecoverableSignature(0x{})", hex::encode(self.to_bytes()))
}
}
#[cfg(test)]
mod tests {
use super::*;
use k256::ecdsa::{signature::hazmat::PrehashSigner, SigningKey};
#[test]
fn bytes_roundtrip() {
let mut bytes = [7u8; SIGNATURE_LEN];
bytes[64] = 1;
let sig = RecoverableSignature::from_bytes(&bytes).unwrap();
assert_eq!(sig.to_bytes(), bytes);
assert_eq!(sig.v(), 1);
}
#[test]
fn normalises_eth_v() {
let mut bytes = [3u8; SIGNATURE_LEN];
bytes[64] = 28;
let sig = RecoverableSignature::from_bytes(&bytes).unwrap();
assert_eq!(sig.v(), 1);
}
#[test]
fn bad_length_and_recid() {
assert!(matches!(
RecoverableSignature::from_bytes(&[0u8; 10]),
Err(SignatureError::BadLength(10))
));
let mut bytes = [0u8; SIGNATURE_LEN];
bytes[64] = 5;
assert!(matches!(
RecoverableSignature::from_bytes(&bytes),
Err(SignatureError::BadRecoveryId(5))
));
}
#[test]
fn from_signature_and_split() {
let signing = SigningKey::from_bytes(&[1u8; 32].into()).unwrap();
let (sig, recid): (Signature, RecoveryId) =
signing.sign_prehash(&[9u8; 32]).unwrap();
let rec = RecoverableSignature::from_signature(&sig, recid);
let (sig2, recid2) = rec.split().unwrap();
assert_eq!(sig, sig2);
assert_eq!(recid.to_byte(), recid2.to_byte());
}
}