use {
crate::{Signer, SignerError},
solana_pubkey::Pubkey,
solana_signature::Signature,
};
pub trait Signers {
fn pubkeys(&self) -> Vec<Pubkey>;
fn try_pubkeys(&self) -> Result<Vec<Pubkey>, SignerError>;
fn sign_message(&self, message: &[u8]) -> Vec<Signature>;
fn try_sign_message(&self, message: &[u8]) -> Result<Vec<Signature>, SignerError>;
fn is_interactive(&self) -> bool;
}
impl<T: ?Sized, S: Signer + ?Sized> Signers for T
where
for<'a> &'a T: IntoIterator<Item = &'a S>,
{
fn pubkeys(&self) -> Vec<Pubkey> {
self.into_iter().map(|keypair| keypair.pubkey()).collect()
}
fn try_pubkeys(&self) -> Result<Vec<Pubkey>, SignerError> {
self.into_iter()
.map(|keypair| keypair.try_pubkey())
.collect()
}
fn sign_message(&self, message: &[u8]) -> Vec<Signature> {
self.into_iter()
.map(|keypair| keypair.sign_message(message))
.collect()
}
fn try_sign_message(&self, message: &[u8]) -> Result<Vec<Signature>, SignerError> {
self.into_iter()
.map(|keypair| keypair.try_sign_message(message))
.collect()
}
fn is_interactive(&self) -> bool {
self.into_iter().any(|s| s.is_interactive())
}
}
#[cfg(test)]
mod tests {
use super::*;
struct Foo;
impl Signer for Foo {
fn try_pubkey(&self) -> Result<Pubkey, SignerError> {
Ok(Pubkey::default())
}
fn try_sign_message(&self, _message: &[u8]) -> Result<Signature, SignerError> {
Ok(Signature::default())
}
fn is_interactive(&self) -> bool {
false
}
}
struct Bar;
impl Signer for Bar {
fn try_pubkey(&self) -> Result<Pubkey, SignerError> {
Ok(Pubkey::default())
}
fn try_sign_message(&self, _message: &[u8]) -> Result<Signature, SignerError> {
Ok(Signature::default())
}
fn is_interactive(&self) -> bool {
false
}
}
#[test]
fn test_dyn_keypairs_compile() {
let xs: Vec<Box<dyn Signer>> = vec![Box::new(Foo {}), Box::new(Bar {})];
assert_eq!(
xs.sign_message(b""),
vec![Signature::default(), Signature::default()],
);
let xs_ref: &[Box<dyn Signer>] = &xs;
assert_eq!(
Signers::sign_message(xs_ref, b""),
vec![Signature::default(), Signature::default()],
);
}
#[test]
fn test_dyn_keypairs_by_ref_compile() {
let foo = Foo {};
let bar = Bar {};
let xs: Vec<&dyn Signer> = vec![&foo, &bar];
assert_eq!(
xs.sign_message(b""),
vec![Signature::default(), Signature::default()],
);
let xs_ref: &[&dyn Signer] = &xs;
assert_eq!(
Signers::sign_message(xs_ref, b""),
vec![Signature::default(), Signature::default()],
);
}
}