1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
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)
    }
}