use std::fmt::Debug;
use openmls_traits::{
signatures::{Signer, SignerError},
storage::{self, StorageProvider, CURRENT_VERSION},
types::{CryptoError, SignatureScheme},
};
use p256::ecdsa::{signature::Signer as P256Signer, Signature, SigningKey};
use rand::rngs::OsRng;
use serde::{Deserialize, Serialize};
use tls_codec::{TlsDeserialize, TlsDeserializeBytes, TlsSerialize, TlsSize};
#[derive(
TlsSerialize, TlsSize, TlsDeserialize, TlsDeserializeBytes, serde::Serialize, serde::Deserialize,
)]
#[cfg_attr(feature = "clonable", derive(Clone))]
pub struct SignatureKeyPair {
private: Vec<u8>,
public: Vec<u8>,
signature_scheme: SignatureScheme,
}
impl Debug for SignatureKeyPair {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("SignatureKeyPair")
.field("private", &"***".to_string())
.field("public", &self.public)
.field("signature_scheme", &self.signature_scheme)
.finish()
}
}
impl Signer for SignatureKeyPair {
fn sign(&self, payload: &[u8]) -> Result<Vec<u8>, SignerError> {
match self.signature_scheme {
SignatureScheme::ECDSA_SECP256R1_SHA256 => {
let k = SigningKey::from_bytes(self.private.as_slice().into())
.map_err(|_| SignerError::SigningError)?;
let signature: Signature = k.sign(payload);
Ok(signature.to_der().to_bytes().into())
}
SignatureScheme::ED25519 => {
let k = ed25519_dalek::SigningKey::try_from(self.private.as_slice())
.map_err(|_| SignerError::SigningError)?;
let signature = k.sign(payload);
Ok(signature.to_bytes().into())
}
_ => Err(SignerError::SigningError),
}
}
fn signature_scheme(&self) -> SignatureScheme {
self.signature_scheme
}
}
fn id(public_key: &[u8], signature_scheme: SignatureScheme) -> Vec<u8> {
const LABEL: &[u8; 22] = b"RustCryptoSignatureKey";
let mut id = public_key.to_vec();
id.extend_from_slice(LABEL);
let signature_scheme = (signature_scheme as u16).to_be_bytes();
id.extend_from_slice(&signature_scheme);
id
}
impl SignatureKeyPair {
pub fn new(signature_scheme: SignatureScheme) -> Result<Self, CryptoError> {
let (private, public) = match signature_scheme {
SignatureScheme::ECDSA_SECP256R1_SHA256 => {
let k = SigningKey::random(&mut OsRng);
let pk = k.verifying_key().to_encoded_point(false).as_bytes().into();
#[allow(deprecated)]
(k.to_bytes().as_slice().into(), pk)
}
SignatureScheme::ED25519 => {
let sk = ed25519_dalek::SigningKey::generate(&mut OsRng);
let pk = sk.verifying_key().to_bytes().into();
(sk.to_bytes().into(), pk)
}
_ => return Err(CryptoError::UnsupportedSignatureScheme),
};
Ok(Self {
private,
public,
signature_scheme,
})
}
pub fn from_raw(signature_scheme: SignatureScheme, private: Vec<u8>, public: Vec<u8>) -> Self {
Self {
private,
public,
signature_scheme,
}
}
pub fn id(&self) -> StorageId {
StorageId {
value: id(&self.public, self.signature_scheme),
}
}
pub fn store<T>(&self, store: &T) -> Result<(), T::Error>
where
T: StorageProvider<CURRENT_VERSION>,
{
store.write_signature_key_pair(&self.id(), self)
}
pub fn read(
store: &impl StorageProvider<CURRENT_VERSION>,
public_key: &[u8],
signature_scheme: SignatureScheme,
) -> Option<Self> {
store
.signature_key_pair(&StorageId {
value: id(public_key, signature_scheme),
})
.ok()
.flatten()
}
pub fn delete<T: StorageProvider<CURRENT_VERSION>>(
store: &T,
public_key: &[u8],
signature_scheme: SignatureScheme,
) -> Result<(), T::Error> {
let id = StorageId {
value: id(public_key, signature_scheme),
};
store.delete_signature_key_pair(&id)
}
pub fn public(&self) -> &[u8] {
self.public.as_ref()
}
pub fn to_public_vec(&self) -> Vec<u8> {
self.public.clone()
}
pub fn signature_scheme(&self) -> SignatureScheme {
self.signature_scheme
}
#[cfg(feature = "test-utils")]
pub fn private(&self) -> &[u8] {
&self.private
}
}
#[derive(Debug, Serialize, Deserialize)]
pub struct StorageId {
value: Vec<u8>,
}
impl From<Vec<u8>> for StorageId {
fn from(vec: Vec<u8>) -> Self {
StorageId { value: vec }
}
}
impl storage::Key<CURRENT_VERSION> for StorageId {}
impl storage::traits::SignaturePublicKey<CURRENT_VERSION> for StorageId {}
impl storage::Entity<CURRENT_VERSION> for SignatureKeyPair {}
impl storage::traits::SignatureKeyPair<CURRENT_VERSION> for SignatureKeyPair {}