98b7c07171
* Update `add_license` script * run script * add `remove duplicate lines script` and run it * Revert changes `English spaces` * strip whitespaces * Revert `GPL` in files with `apache/mit license` * don't append `gpl license` in files with other lic * Don't append `gpl header` in files with other lic. * re-ran script * include c and cpp files too * remove duplicate header * rebase nit
271 lines
8.0 KiB
Rust
271 lines
8.0 KiB
Rust
// Copyright 2015-2018 Parity Technologies (UK) Ltd.
|
|
// This file is part of Parity.
|
|
|
|
// Parity is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
|
|
// Parity is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
//! Encryption schemes supported by RPC layer.
|
|
|
|
use crypto::aes_gcm::{Encryptor, Decryptor};
|
|
use ethkey::crypto::ecies;
|
|
use ethereum_types::H256;
|
|
use ethkey::{self, Public, Secret};
|
|
use mem::Memzero;
|
|
|
|
/// Length of AES key
|
|
pub const AES_KEY_LEN: usize = 32;
|
|
/// Length of AES nonce (IV)
|
|
pub const AES_NONCE_LEN: usize = 12;
|
|
|
|
// nonce used for encryption when broadcasting
|
|
const BROADCAST_IV: [u8; AES_NONCE_LEN] = [0xff; AES_NONCE_LEN];
|
|
|
|
// how to encode aes key/nonce.
|
|
enum AesEncode {
|
|
AppendedNonce, // receiver known, random nonce appended.
|
|
OnTopics(Vec<H256>), // receiver knows topics but not key. nonce global.
|
|
}
|
|
|
|
enum EncryptionInner {
|
|
AES(Memzero<[u8; AES_KEY_LEN]>, [u8; AES_NONCE_LEN], AesEncode),
|
|
ECIES(Public),
|
|
}
|
|
|
|
/// Encryption good for single usage.
|
|
pub struct EncryptionInstance(EncryptionInner);
|
|
|
|
impl EncryptionInstance {
|
|
/// ECIES encryption using public key. Fails if invalid public key.
|
|
pub fn ecies(public: Public) -> Result<Self, &'static str> {
|
|
if !ethkey::public_is_valid(&public) {
|
|
return Err("Invalid public key");
|
|
}
|
|
|
|
Ok(EncryptionInstance(EncryptionInner::ECIES(public)))
|
|
}
|
|
|
|
/// 256-bit AES GCM encryption with given nonce.
|
|
/// It is extremely insecure to reuse nonces.
|
|
///
|
|
/// If generating nonces with a secure RNG, limit uses such that
|
|
/// the chance of collision is negligible.
|
|
pub fn aes(key: Memzero<[u8; AES_KEY_LEN]>, nonce: [u8; AES_NONCE_LEN]) -> Self {
|
|
EncryptionInstance(EncryptionInner::AES(key, nonce, AesEncode::AppendedNonce))
|
|
}
|
|
|
|
/// Broadcast encryption for the message based on the given topics.
|
|
///
|
|
/// Key reuse here is extremely dangerous. It should be randomly generated
|
|
/// with a secure RNG.
|
|
pub fn broadcast(key: Memzero<[u8; AES_KEY_LEN]>, topics: Vec<H256>) -> Self {
|
|
EncryptionInstance(EncryptionInner::AES(key, BROADCAST_IV, AesEncode::OnTopics(topics)))
|
|
}
|
|
|
|
/// Encrypt the supplied plaintext
|
|
pub fn encrypt(self, plain: &[u8]) -> Option<Vec<u8>> {
|
|
match self.0 {
|
|
EncryptionInner::AES(key, nonce, encode) => {
|
|
match encode {
|
|
AesEncode::AppendedNonce => {
|
|
let mut enc = Encryptor::aes_256_gcm(&*key).ok()?;
|
|
let mut buf = enc.encrypt(&nonce, plain.to_vec()).ok()?;
|
|
buf.extend(&nonce[..]);
|
|
Some(buf)
|
|
}
|
|
AesEncode::OnTopics(topics) => {
|
|
let mut buf = Vec::new();
|
|
for mut t in topics {
|
|
xor(&mut t.0, &key);
|
|
buf.extend(&t.0);
|
|
}
|
|
let mut enc = Encryptor::aes_256_gcm(&*key).ok()?;
|
|
enc.offset(buf.len());
|
|
buf.extend(plain);
|
|
let ciphertext = enc.encrypt(&nonce, buf).ok()?;
|
|
Some(ciphertext)
|
|
}
|
|
}
|
|
}
|
|
EncryptionInner::ECIES(valid_public) => {
|
|
ecies::encrypt(&valid_public, &[], plain).ok()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#[inline]
|
|
fn xor(a: &mut [u8; 32], b: &[u8; 32]) {
|
|
for i in 0 .. 32 {
|
|
a[i] ^= b[i]
|
|
}
|
|
}
|
|
|
|
enum AesExtract {
|
|
AppendedNonce(Memzero<[u8; AES_KEY_LEN]>), // extract appended nonce.
|
|
OnTopics(usize, usize, H256), // number of topics, index we know, topic we know.
|
|
}
|
|
|
|
enum DecryptionInner {
|
|
AES(AesExtract),
|
|
ECIES(Secret),
|
|
}
|
|
|
|
/// Decryption instance good for single usage.
|
|
pub struct DecryptionInstance(DecryptionInner);
|
|
|
|
impl DecryptionInstance {
|
|
/// ECIES decryption using secret key. Fails if invalid secret.
|
|
pub fn ecies(secret: Secret) -> Result<Self, &'static str> {
|
|
secret.check_validity().map_err(|_| "Invalid secret key")?;
|
|
|
|
Ok(DecryptionInstance(DecryptionInner::ECIES(secret)))
|
|
}
|
|
|
|
/// 256-bit AES GCM decryption with appended nonce.
|
|
pub fn aes(key: Memzero<[u8; AES_KEY_LEN]>) -> Self {
|
|
DecryptionInstance(DecryptionInner::AES(AesExtract::AppendedNonce(key)))
|
|
}
|
|
|
|
/// Decode broadcast based on number of topics and known topic.
|
|
/// Known topic index may not be larger than num topics - 1.
|
|
pub fn broadcast(num_topics: usize, topic_idx: usize, known_topic: H256) -> Result<Self, &'static str> {
|
|
if topic_idx >= num_topics { return Err("topic index out of bounds") }
|
|
|
|
Ok(DecryptionInstance(DecryptionInner::AES(AesExtract::OnTopics(num_topics, topic_idx, known_topic))))
|
|
}
|
|
|
|
/// Decrypt ciphertext. Fails if it's an invalid message.
|
|
pub fn decrypt(self, ciphertext: &[u8]) -> Option<Vec<u8>> {
|
|
match self.0 {
|
|
DecryptionInner::AES(extract) => {
|
|
match extract {
|
|
AesExtract::AppendedNonce(key) => {
|
|
if ciphertext.len() < AES_NONCE_LEN {
|
|
return None
|
|
}
|
|
// nonce is the suffix of ciphertext.
|
|
let mut nonce = [0; AES_NONCE_LEN];
|
|
let nonce_offset = ciphertext.len() - AES_NONCE_LEN;
|
|
nonce.copy_from_slice(&ciphertext[nonce_offset..]);
|
|
Decryptor::aes_256_gcm(&*key).ok()?
|
|
.decrypt(&nonce, Vec::from(&ciphertext[..nonce_offset]))
|
|
.ok()
|
|
}
|
|
AesExtract::OnTopics(num_topics, known_index, known_topic) => {
|
|
if ciphertext.len() < num_topics * 32 {
|
|
return None
|
|
}
|
|
let mut salted_topic = H256::new();
|
|
salted_topic.copy_from_slice(&ciphertext[(known_index * 32)..][..32]);
|
|
let key = Memzero::from((salted_topic ^ known_topic).0);
|
|
let offset = num_topics * 32;
|
|
Decryptor::aes_256_gcm(&*key).ok()?
|
|
.decrypt(&BROADCAST_IV, Vec::from(&ciphertext[offset..]))
|
|
.ok()
|
|
}
|
|
}
|
|
}
|
|
DecryptionInner::ECIES(secret) => {
|
|
// secret is checked for validity, so only fails on invalid message.
|
|
ecies::decrypt(&secret, &[], ciphertext).ok()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn encrypt_asymmetric() {
|
|
use ethkey::{Generator, Random};
|
|
|
|
let key_pair = Random.generate().unwrap();
|
|
let test_message = move |message: &[u8]| {
|
|
let instance = EncryptionInstance::ecies(key_pair.public().clone()).unwrap();
|
|
let ciphertext = instance.encrypt(&message).unwrap();
|
|
|
|
if !message.is_empty() {
|
|
assert!(&ciphertext[..message.len()] != message)
|
|
}
|
|
|
|
let instance = DecryptionInstance::ecies(key_pair.secret().clone()).unwrap();
|
|
let decrypted = instance.decrypt(&ciphertext).unwrap();
|
|
|
|
assert_eq!(message, &decrypted[..])
|
|
};
|
|
|
|
test_message(&[1, 2, 3, 4, 5]);
|
|
test_message(&[]);
|
|
test_message(&[255; 512]);
|
|
}
|
|
|
|
#[test]
|
|
fn encrypt_symmetric() {
|
|
use rand::{Rng, OsRng};
|
|
|
|
let mut rng = OsRng::new().unwrap();
|
|
let mut test_message = move |message: &[u8]| {
|
|
let key = Memzero::from(rng.gen::<[u8; 32]>());
|
|
|
|
let instance = EncryptionInstance::aes(key.clone(), rng.gen());
|
|
let ciphertext = instance.encrypt(message).unwrap();
|
|
|
|
if !message.is_empty() {
|
|
assert!(&ciphertext[..message.len()] != message)
|
|
}
|
|
|
|
let instance = DecryptionInstance::aes(key);
|
|
let decrypted = instance.decrypt(&ciphertext).unwrap();
|
|
|
|
assert_eq!(message, &decrypted[..])
|
|
};
|
|
|
|
test_message(&[1, 2, 3, 4, 5]);
|
|
test_message(&[]);
|
|
test_message(&[255; 512]);
|
|
}
|
|
|
|
#[test]
|
|
fn encrypt_broadcast() {
|
|
use rand::{Rng, OsRng};
|
|
|
|
let mut rng = OsRng::new().unwrap();
|
|
|
|
let mut test_message = move |message: &[u8]| {
|
|
let all_topics = (0..5).map(|_| rng.gen()).collect::<Vec<_>>();
|
|
let known_idx = 2;
|
|
let known_topic = all_topics[2];
|
|
let key = Memzero::from(rng.gen::<[u8; 32]>());
|
|
|
|
let instance = EncryptionInstance::broadcast(key, all_topics);
|
|
let ciphertext = instance.encrypt(message).unwrap();
|
|
|
|
if !message.is_empty() {
|
|
assert!(&ciphertext[..message.len()] != message)
|
|
}
|
|
|
|
let instance = DecryptionInstance::broadcast(5, known_idx, known_topic).unwrap();
|
|
|
|
let decrypted = instance.decrypt(&ciphertext).unwrap();
|
|
|
|
assert_eq!(message, &decrypted[..])
|
|
};
|
|
|
|
test_message(&[1, 2, 3, 4, 5]);
|
|
test_message(&[]);
|
|
test_message(&[255; 512]);
|
|
}
|
|
}
|