Plume/src/activity_pub/sign.rs

58 lines
1.6 KiB
Rust
Raw Normal View History

2018-04-29 17:40:10 +02:00
use base64;
2018-05-19 09:39:59 +02:00
use chrono::Utc;
use diesel::PgConnection;
2018-04-29 17:40:10 +02:00
use hex;
2018-05-19 09:39:59 +02:00
use openssl::{
pkey::PKey,
rsa::Rsa,
sha::sha256
};
2018-04-29 17:40:10 +02:00
use serde_json;
/// Returns (public key, private key)
pub fn gen_keypair() -> (Vec<u8>, Vec<u8>) {
let keypair = Rsa::generate(2048).unwrap();
let keypair = PKey::from_rsa(keypair).unwrap();
(keypair.public_key_to_pem().unwrap(), keypair.private_key_to_pem_pkcs8().unwrap())
}
2018-04-29 17:40:10 +02:00
pub trait Signer {
fn get_key_id(&self, conn: &PgConnection) -> String;
2018-04-29 17:40:10 +02:00
/// Sign some data with the signer keypair
fn sign(&self, to_sign: String) -> Vec<u8>;
2018-04-29 17:40:10 +02:00
}
pub trait Signable {
fn sign<T>(&mut self, creator: &T, conn: &PgConnection) -> &mut Self where T: Signer;
2018-04-29 17:40:10 +02:00
fn hash(data: String) -> String {
let bytes = data.into_bytes();
hex::encode(sha256(&bytes[..]))
}
}
impl Signable for serde_json::Value {
fn sign<T: Signer>(&mut self, creator: &T, conn: &PgConnection) -> &mut serde_json::Value {
let creation_date = Utc::now().to_rfc3339();
2018-04-29 17:40:10 +02:00
let mut options = json!({
"type": "RsaSignature2017",
"creator": creator.get_key_id(conn),
"created": creation_date
2018-04-29 17:40:10 +02:00
});
let options_hash = Self::hash(json!({
"@context": "https://w3id.org/identity/v1",
"created": creation_date
}).to_string());
2018-04-29 17:40:10 +02:00
let document_hash = Self::hash(self.to_string());
let to_be_signed = options_hash + &document_hash;
let signature = base64::encode(&creator.sign(to_be_signed));
options["signaureValue"] = serde_json::Value::String(signature);
self["signature"] = options;
self
}
}