crypto: Don't create a new salt when we just want to rederive a recovery key

This commit is contained in:
Damir Jelić 2021-11-11 16:00:35 +01:00
parent f1da5a1c7c
commit 7cb143e970
4 changed files with 90 additions and 32 deletions

View File

@ -120,7 +120,7 @@ internal class RustKeyBackupService @Inject constructor(
runCatching { runCatching {
withContext(coroutineDispatchers.crypto) { withContext(coroutineDispatchers.crypto) {
val key = if (password != null) { val key = if (password != null) {
BackupRecoveryKey.fromPassphrase(password) BackupRecoveryKey.newFromPassphrase(password)
} else { } else {
BackupRecoveryKey() BackupRecoveryKey()
} }
@ -397,9 +397,15 @@ internal class RustKeyBackupService @Inject constructor(
callback: MatrixCallback<Unit>) { callback: MatrixCallback<Unit>) {
cryptoCoroutineScope.launch { cryptoCoroutineScope.launch {
try { try {
val key = BackupRecoveryKey.fromPassphrase(password) val key = recoveryKeyFromPassword(password, keysBackupVersion)
checkRecoveryKey(key, keysBackupVersion)
trustKeysBackupVersion(keysBackupVersion, true, callback) if (key == null) {
Timber.w("trustKeysBackupVersionWithPassphrase: Key backup is missing required data")
callback.onFailure(IllegalArgumentException("Missing element"))
} else {
checkRecoveryKey(key, keysBackupVersion)
trustKeysBackupVersion(keysBackupVersion, true, callback)
}
} catch (exception: Throwable) { } catch (exception: Throwable) {
callback.onFailure(exception) callback.onFailure(exception)
} }
@ -589,7 +595,15 @@ internal class RustKeyBackupService @Inject constructor(
cryptoCoroutineScope.launch(coroutineDispatchers.main) { cryptoCoroutineScope.launch(coroutineDispatchers.main) {
runCatching { runCatching {
val recoveryKey = withContext(coroutineDispatchers.crypto) { val recoveryKey = withContext(coroutineDispatchers.crypto) {
BackupRecoveryKey.fromPassphrase(password) val key = recoveryKeyFromPassword(password, keysBackupVersion)
if (key == null) {
Timber.w("trustKeysBackupVersionWithPassphrase: Key backup is missing required data")
throw IllegalArgumentException("Missing element")
} else {
key
}
} }
restoreBackup(keysBackupVersion, recoveryKey, roomId, sessionId, stepProgressListener) restoreBackup(keysBackupVersion, recoveryKey, roomId, sessionId, stepProgressListener)
@ -749,6 +763,33 @@ internal class RustKeyBackupService @Inject constructor(
return SavedKeyBackupKeyInfo(info.recoveryKey, info.backupVersion) return SavedKeyBackupKeyInfo(info.recoveryKey, info.backupVersion)
} }
/**
* Compute the recovery key from a password and key backup version.
*
* @param password the password.
* @param keysBackupData the backup and its auth data.
*
* @return the recovery key if successful, null in other cases
*/
@WorkerThread
private fun recoveryKeyFromPassword(password: String, keysBackupData: KeysVersionResult): BackupRecoveryKey? {
val authData = getMegolmBackupAuthData(keysBackupData)
if (authData == null) {
Timber.w("recoveryKeyFromPassword: invalid parameter")
return null
}
if (authData.privateKeySalt.isNullOrBlank()
|| authData.privateKeyIterations == null) {
Timber.w("recoveryKeyFromPassword: Salt and/or iterations not found in key backup auth data")
return null
}
return BackupRecoveryKey.fromPassphrase(password, authData.privateKeySalt, authData.privateKeyIterations)
}
/** /**
* Extract MegolmBackupAuthData data from a backup version. * Extract MegolmBackupAuthData data from a backup version.
* *

View File

@ -5,47 +5,48 @@ use sha2::Sha512;
use std::{collections::HashMap, iter}; use std::{collections::HashMap, iter};
use thiserror::Error; use thiserror::Error;
use matrix_sdk_crypto::backups::{RecoveryKey, OlmPkDecryptionError}; use matrix_sdk_crypto::backups::{OlmPkDecryptionError, RecoveryKey};
/// TODO /// The private part of the backup key, the one used for recovery.
pub struct BackupRecoveryKey { pub struct BackupRecoveryKey {
pub(crate) inner: RecoveryKey, pub(crate) inner: RecoveryKey,
passphrase_info: Option<PassphraseInfo>, passphrase_info: Option<PassphraseInfo>,
} }
/// TODO /// Error type for the decryption of backed up room keys.
#[derive(Debug, Error)] #[derive(Debug, Error)]
pub enum PkDecryptionError { pub enum PkDecryptionError {
/// TODO /// An internal libolm error happened during decryption.
#[error("Error decryption a PkMessage {0}")] #[error("Error decryption a PkMessage {0}")]
Olm(#[from] OlmPkDecryptionError), Olm(#[from] OlmPkDecryptionError),
} }
/// TODO /// Struct containing info about the way the backup key got derived from a
/// passphrase.
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
pub struct PassphraseInfo { pub struct PassphraseInfo {
/// TODO /// The salt that was used during key derivation.
pub private_key_salt: String, pub private_key_salt: String,
/// TODO /// The number of PBKDF rounds that were used for key derivation.
pub private_key_iterations: i32, pub private_key_iterations: i32,
} }
/// TODO /// The public part of the backup key.
pub struct BackupKey { pub struct BackupKey {
/// TODO /// The actuall base64 encoded public key.
pub public_key: String, pub public_key: String,
/// TODO /// Signatures that have signed our backup key.
pub signatures: HashMap<String, HashMap<String, String>>, pub signatures: HashMap<String, HashMap<String, String>>,
/// TODO /// The passphrase info, if the key was derived from one.
pub passphrase_info: Option<PassphraseInfo>, pub passphrase_info: Option<PassphraseInfo>,
} }
impl BackupRecoveryKey { impl BackupRecoveryKey {
const KEY_SIZE: usize = 32; const KEY_SIZE: usize = 32;
const SALT_SIZE: usize = 32; const SALT_SIZE: usize = 32;
const PBKDF_ROUNDS: u32 = 500_000; const PBKDF_ROUNDS: i32 = 500_000;
/// TODO /// Create a new random [`BackupRecoveryKey`].
pub fn new() -> Self { pub fn new() -> Self {
Self { Self {
inner: RecoveryKey::new() inner: RecoveryKey::new()
@ -54,26 +55,26 @@ impl BackupRecoveryKey {
} }
} }
/// TODO /// Try to create a [`BackupRecoveryKey`] from a base 64 encoded string.
pub fn from_base64(key: String) -> Self { pub fn from_base64(key: String) -> Self {
Self { Self {
// TODO remove the unwrap
inner: RecoveryKey::from_base64(&key).unwrap(), inner: RecoveryKey::from_base64(&key).unwrap(),
passphrase_info: None, passphrase_info: None,
} }
} }
/// TODO /// Try to create a [`BackupRecoveryKey`] from a base 58 encoded string.
pub fn from_base58(key: String) -> Self { pub fn from_base58(key: String) -> Self {
Self { Self {
// TODO remove the unwrap
inner: RecoveryKey::from_base58(&key).unwrap(), inner: RecoveryKey::from_base58(&key).unwrap(),
passphrase_info: None, passphrase_info: None,
} }
} }
/// TODO /// Create a new [`BackupRecoveryKey`] from the given passphrase.
pub fn from_passphrase(passphrase: String) -> Self { pub fn new_from_passphrase(passphrase: String) -> Self {
let mut key = [0u8; Self::KEY_SIZE];
let mut rng = thread_rng(); let mut rng = thread_rng();
let salt: String = iter::repeat(()) let salt: String = iter::repeat(())
.map(|()| rng.sample(Alphanumeric)) .map(|()| rng.sample(Alphanumeric))
@ -81,10 +82,18 @@ impl BackupRecoveryKey {
.take(Self::SALT_SIZE) .take(Self::SALT_SIZE)
.collect(); .collect();
Self::from_passphrase(passphrase, salt, Self::PBKDF_ROUNDS)
}
/// Restore a [`BackupRecoveryKey`] from the given passphrase.
pub fn from_passphrase(passphrase: String, salt: String, rounds: i32) -> Self {
let mut key = [0u8; Self::KEY_SIZE];
let rounds = rounds as u32;
pbkdf2::<Hmac<Sha512>>( pbkdf2::<Hmac<Sha512>>(
passphrase.as_bytes(), passphrase.as_bytes(),
salt.as_bytes(), salt.as_bytes(),
Self::PBKDF_ROUNDS, rounds,
&mut key, &mut key,
); );
@ -92,12 +101,12 @@ impl BackupRecoveryKey {
inner: RecoveryKey::from_bytes(key), inner: RecoveryKey::from_bytes(key),
passphrase_info: Some(PassphraseInfo { passphrase_info: Some(PassphraseInfo {
private_key_salt: salt, private_key_salt: salt,
private_key_iterations: Self::PBKDF_ROUNDS as i32, private_key_iterations: rounds as i32,
}), }),
} }
} }
/// TODO /// Get the public part of the backup key.
pub fn public_key(&self) -> BackupKey { pub fn public_key(&self) -> BackupKey {
let public_key = self.inner.public_key(); let public_key = self.inner.public_key();
@ -113,22 +122,24 @@ impl BackupRecoveryKey {
.collect(); .collect();
BackupKey { BackupKey {
public_key: public_key.encoded_key(), public_key: public_key.to_base64(),
signatures, signatures,
passphrase_info: self.passphrase_info.clone(), passphrase_info: self.passphrase_info.clone(),
} }
} }
/// TODO /// Convert the recovery key to a base 58 encoded string.
pub fn to_base58(&self) -> String { pub fn to_base58(&self) -> String {
self.inner.to_base58() self.inner.to_base58()
} }
/// TODO /// Convert the recovery key to a base 64 encoded string.
pub fn to_base64(&self) -> String { pub fn to_base64(&self) -> String {
self.inner.to_base64() self.inner.to_base64()
} }
/// Try to decrypt a message that was encrypted using the public part of the
/// backup key.
pub fn decrypt( pub fn decrypt(
&self, &self,
ephemeral_key: String, ephemeral_key: String,

View File

@ -1311,7 +1311,11 @@ impl OlmMachine {
key: Option<String>, key: Option<String>,
version: Option<String>, version: Option<String>,
) -> Result<(), CryptoStoreError> { ) -> Result<(), CryptoStoreError> {
let key = key.map(|k| RecoveryKey::from_base64(&k)).transpose().ok().flatten(); let key = key
.map(|k| RecoveryKey::from_base64(&k))
.transpose()
.ok()
.flatten();
Ok(self Ok(self
.runtime .runtime
.block_on(self.inner.backup_machine().save_recovery_key(key, version))?) .block_on(self.inner.backup_machine().save_recovery_key(key, version))?)

View File

@ -400,7 +400,9 @@ interface BackupRecoveryKey {
[Name=from_base64] [Name=from_base64]
constructor(string key); constructor(string key);
[Name=from_passphrase] [Name=from_passphrase]
constructor(string key); constructor(string passphrase, string salt, i32 rounds);
[Name=new_from_passphrase]
constructor(string passphrase);
[Name=from_base58] [Name=from_base58]
constructor(string key); constructor(string key);
string to_base58(); string to_base58();