-
Notifications
You must be signed in to change notification settings - Fork 33
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
WIP: move MasterKey and StorageServiceKey to libsignal-service-rs
- Loading branch information
Showing
3 changed files
with
116 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,111 @@ | ||
const MASTER_KEY_LEN: usize = 32; | ||
const STORAGE_KEY_LEN: usize = 32; | ||
|
||
#[derive(Debug, PartialEq)] | ||
pub struct MasterKey { | ||
pub inner: [u8; MASTER_KEY_LEN], | ||
} | ||
|
||
impl MasterKey { | ||
pub fn generate() -> Self { | ||
use rand::Rng; | ||
|
||
// Create random bytes | ||
let mut rng = rand::thread_rng(); | ||
let mut inner = [0_u8; MASTER_KEY_LEN]; | ||
rng.fill(&mut inner); | ||
Self { inner } | ||
} | ||
|
||
pub fn from_slice( | ||
slice: &Vec<u8>, | ||
) -> Result<Self, std::array::TryFromSliceError> { | ||
let inner = slice.as_slice().try_into()?; | ||
Ok(Self { inner }) | ||
} | ||
} | ||
|
||
impl From<MasterKey> for Vec<u8> { | ||
fn from(val: MasterKey) -> Self { | ||
val.inner.to_vec() | ||
} | ||
} | ||
|
||
#[derive(Debug, PartialEq)] | ||
pub struct StorageServiceKey { | ||
pub inner: [u8; STORAGE_KEY_LEN], | ||
} | ||
|
||
impl StorageServiceKey { | ||
pub fn from_master_key(master_key: &MasterKey) -> Self { | ||
use hmac::{Hmac, Mac}; | ||
use sha2::Sha256; | ||
|
||
type HmacSha256 = Hmac<Sha256>; | ||
const KEY: &[u8] = b"Storage Service Encryption"; | ||
|
||
let mut mac = HmacSha256::new_from_slice(&master_key.inner).unwrap(); | ||
mac.update(KEY); | ||
let result = mac.finalize(); | ||
let inner: [u8; STORAGE_KEY_LEN] = result.into_bytes().into(); | ||
|
||
Self { inner } | ||
} | ||
|
||
pub fn from_slice( | ||
slice: &Vec<u8>, | ||
) -> Result<Self, std::array::TryFromSliceError> { | ||
let inner = slice.as_slice().try_into()?; | ||
Ok(Self { inner }) | ||
} | ||
} | ||
|
||
impl From<StorageServiceKey> for Vec<u8> { | ||
fn from(val: StorageServiceKey) -> Self { | ||
val.inner.to_vec() | ||
} | ||
} | ||
|
||
/// Storage trait for handling MasterKey and StorageKey. | ||
pub trait MasterKeyStore { | ||
/// Fetch the master key from the store if it exists. | ||
fn fetch_master_key(&self) -> Option<MasterKey>; | ||
|
||
/// Fetch the storage service key from the store if it exists. | ||
fn fetch_storage_service_key(&self) -> Option<StorageServiceKey>; | ||
|
||
/// Save (or clear) the master key to the store. | ||
fn store_master_key(&self, master_key: Option<&MasterKey>); | ||
|
||
/// Save (or clear) the storage service key to the store. | ||
fn store_storage_service_key( | ||
&self, | ||
storage_key: Option<&StorageServiceKey>, | ||
); | ||
} | ||
|
||
mod tests { | ||
#[test] | ||
fn derive_storage_key_from_master_key() { | ||
use super::{MasterKey, StorageServiceKey}; | ||
use base64::prelude::*; | ||
|
||
// This test passed with actual 'masterKey' and 'storageKey' values taken | ||
// from Signal Desktop v7.23.0 database at 2024-09-08 after linking it with Signal Andoid. | ||
|
||
let master_key_bytes = BASE64_STANDARD | ||
.decode("9hquLIIZmom8fHF7H8pbUAreawmPLEqli5ceJ94pFkU=") | ||
.unwrap(); | ||
let storage_key_bytes = BASE64_STANDARD | ||
.decode("QMgZ5RGTLFTr4u/J6nypaJX6DKDlSgMw8vmxU6gxnvI=") | ||
.unwrap(); | ||
assert_eq!(master_key_bytes.len(), 32); | ||
assert_eq!(storage_key_bytes.len(), 32); | ||
|
||
let master_key = MasterKey::from_slice(&master_key_bytes).unwrap(); | ||
let storage_key = StorageServiceKey::from_master_key(&master_key); | ||
|
||
assert_eq!(master_key.inner, master_key_bytes.as_slice()); | ||
assert_eq!(storage_key.inner, storage_key_bytes.as_slice()); | ||
} | ||
} |