use signature::Error as SignatureError;
use crate::signing;
use crate::{ByteVisitor, VisitBytes};
pub trait Encode {
fn encode(&self) -> Vec<u8>;
}
#[derive(Default)]
struct EncodingVisitor {
bytes: Vec<u8>,
}
impl ByteVisitor for EncodingVisitor {
fn visit_bytes(&mut self, bytes: impl AsRef<[u8]>) {
self.bytes.extend(bytes.as_ref())
}
}
impl<T> Encode for T
where
T: VisitBytes,
{
fn encode(&self) -> Vec<u8> {
let mut visitor = EncodingVisitor::default();
self.visit(&mut visitor);
visitor.bytes
}
}
pub trait Signable: Encode {
const PREFIX: &'static [u8];
fn sign(
&self,
private_key: &signing::PrivateKey,
) -> Result<signing::Signature, SignatureError> {
let prefixed_content = [Self::PREFIX, b":", self.encode().as_slice()].concat();
private_key.sign(&prefixed_content)
}
fn verify(
public_key: &signing::PublicKey,
msg: &[u8],
signature: &signing::Signature,
) -> Result<(), SignatureError> {
let prefixed_content = [Self::PREFIX, b":", msg].concat();
public_key.verify(&prefixed_content, signature)
}
}