#![cfg(any(test, feature = "testutils"))]
pub trait Sign<MSG> {
type Signature;
type Error;
fn sign(&self, m: MSG) -> Result<Self::Signature, Self::Error>;
}
pub mod ed25519 {
use crate::xdr;
use xdr::{Limited, Limits, WriteXdr};
#[derive(Debug)]
pub enum Error<E: std::error::Error> {
XdrError(xdr::Error),
Ed25519SignatureError(ed25519_dalek::SignatureError),
ConversionError(E),
}
impl<E: std::error::Error> std::error::Error for Error<E> {
#[must_use]
fn source(&self) -> Option<&(dyn std::error::Error + 'static)> {
match self {
Self::XdrError(e) => e.source(),
Self::Ed25519SignatureError(e) => e.source(),
Self::ConversionError(e) => e.source(),
}
}
}
impl<E: std::error::Error> std::fmt::Display for Error<E> {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
match self {
Self::XdrError(e) => write!(f, "{}", e),
Self::Ed25519SignatureError(e) => write!(f, "{}", e),
Self::ConversionError(e) => write!(f, "{}", e),
}
}
}
impl<E: std::error::Error> From<xdr::Error> for Error<E> {
fn from(e: xdr::Error) -> Self {
Error::XdrError(e)
}
}
impl<E: std::error::Error> From<ed25519_dalek::SignatureError> for Error<E> {
fn from(e: ed25519_dalek::SignatureError) -> Self {
Error::Ed25519SignatureError(e)
}
}
pub use super::Sign;
impl<S, M> Sign<M> for S
where
S: ed25519_dalek::Signer<ed25519_dalek::Signature>,
M: TryInto<xdr::ScVal>,
<M as TryInto<xdr::ScVal>>::Error: std::error::Error,
{
type Error = Error<<M as TryInto<xdr::ScVal>>::Error>;
type Signature = [u8; 64];
fn sign(&self, m: M) -> Result<Self::Signature, Self::Error> {
let mut buf = Vec::<u8>::new();
let val: xdr::ScVal = m.try_into().map_err(Self::Error::ConversionError)?;
val.write_xdr(&mut Limited::new(&mut buf, Limits::none()))?;
Ok(self.try_sign(&buf)?.to_bytes())
}
}
#[cfg(test)]
mod test {
use super::Sign;
use ed25519_dalek::SigningKey;
#[test]
fn sign() {
let sk = SigningKey::from_bytes(
&hex::decode("5acc7253295dfc356c046297925a369f3d2762d00afdf2583ecbe92180b07c37")
.unwrap()
.try_into()
.unwrap(),
);
let sig = sk.sign(128i64).unwrap();
assert_eq!(
hex::encode(sig),
"a9b9dfac10bc1e5c8bc565e9515e5d086e3264b71bf4daf2c7340e1d10fae86e2563fa1d639ff153559a9710dfa270a9462fe87faa0e18a7a54a8a1a6151e909",
);
}
}
}