sidetree-client 0.1.2

Client library for managing DID operations for Sidetree API services
Documentation
use secp256k1::{Message, PublicKey, RecoveryId, SecretKey, Signature};

use crate::{
    did::{JsonWebKey, Purpose},
    encoder::encode,
};

#[derive(Debug, Clone)]
pub struct KeyPair {
    public_key: PublicKey,
    secret_key: Option<SecretKey>,
}

impl KeyPair {
    pub fn sign(&self, message: &[u8]) -> (Signature, RecoveryId) {
        secp256k1::sign(
            &Message::parse_slice(message).unwrap(),
            &self.secret_key.as_ref().unwrap(),
        )
    }

    pub fn verify(&self, message: &[u8], signature: &[u8]) -> bool {
        secp256k1::verify(
            &Message::parse_slice(message).unwrap(),
            &Signature::parse_slice(signature).unwrap(),
            &self.public_key,
        )
    }

    pub fn to_public_key(&self, id: String, purposes: Option<Purpose>) -> crate::PublicKey {
        let mut jwk: JsonWebKey = self.into();
        jwk.d = None;

        crate::PublicKey {
            id,
            key_type: "EcdsaSecp256k1VerificationKey2019".to_string(),
            purposes,
            jwk: Some(jwk),
        }
    }

    pub fn random() -> KeyPair {
        let mut seed = [0u8; 32];
        getrandom::getrandom(&mut seed).expect("couldn't generate random seed");

        // TODO: Add rng support
        let secret_key = SecretKey::parse(&seed).unwrap();
        let public_key = PublicKey::from_secret_key(&secret_key);

        KeyPair {
            public_key,
            secret_key: Some(secret_key),
        }
    }
}

impl From<&KeyPair> for JsonWebKey {
    fn from(keypair: &KeyPair) -> Self {
        let serialized_public_key = keypair.public_key.serialize();

        JsonWebKey {
            key_type: "EC".into(),
            curve: "secp256k1".into(),
            x: encode(serialized_public_key[1..33].as_ref()),
            y: encode(serialized_public_key[33..65].as_ref()),
            d: keypair
                .secret_key
                .as_ref()
                .map(|secret_key| encode(secret_key.serialize())),
        }
    }
}

#[cfg(test)]
mod test {
    #[test]
    fn generate_random_key() {
        let keypair = super::KeyPair::random();

        assert!(matches!(keypair.secret_key, Some(_)));
    }
}