mirror of
https://github.com/str4d/rage.git
synced 2025-04-03 19:07:42 +03:00
age: Make recipients from encrypted identities more efficient
We now merge plugin recipients together, so we only run each plugin once during encryption.
This commit is contained in:
parent
52fd675bbd
commit
8dcdacc1ac
2 changed files with 22 additions and 36 deletions
|
@ -2,9 +2,7 @@
|
|||
|
||||
use std::{cell::Cell, io};
|
||||
|
||||
use crate::{
|
||||
fl, scrypt, Callbacks, DecryptError, Decryptor, EncryptError, IdentityFile, IdentityFileEntry,
|
||||
};
|
||||
use crate::{fl, scrypt, Callbacks, DecryptError, Decryptor, EncryptError, IdentityFile};
|
||||
|
||||
/// The state of the encrypted age identity.
|
||||
enum IdentityState<R: io::Read> {
|
||||
|
@ -12,7 +10,7 @@ enum IdentityState<R: io::Read> {
|
|||
decryptor: Decryptor<R>,
|
||||
max_work_factor: Option<u8>,
|
||||
},
|
||||
Decrypted(Vec<IdentityFileEntry>),
|
||||
Decrypted(IdentityFile),
|
||||
|
||||
/// The file was not correctly encrypted, or did not contain age identities. We cache
|
||||
/// this error in case the caller tries to use this identity again. The `Option` is to
|
||||
|
@ -36,7 +34,7 @@ impl<R: io::Read> IdentityState<R> {
|
|||
self,
|
||||
filename: Option<&str>,
|
||||
callbacks: C,
|
||||
) -> Result<(Vec<IdentityFileEntry>, bool), DecryptError> {
|
||||
) -> Result<(IdentityFile, bool), DecryptError> {
|
||||
match self {
|
||||
Self::Encrypted {
|
||||
decryptor,
|
||||
|
@ -66,10 +64,10 @@ impl<R: io::Read> IdentityState<R> {
|
|||
})
|
||||
.and_then(|stream| {
|
||||
let file = IdentityFile::from_buffer(io::BufReader::new(stream))?;
|
||||
Ok((file.into_identities(), true))
|
||||
Ok((file, true))
|
||||
})
|
||||
}
|
||||
Self::Decrypted(identities) => Ok((identities, false)),
|
||||
Self::Decrypted(identity_file) => Ok((identity_file, false)),
|
||||
// `IdentityState::decrypt` is only ever called with `Some`.
|
||||
Self::Poisoned(e) => Err(e.unwrap()),
|
||||
}
|
||||
|
@ -117,13 +115,9 @@ impl<R: io::Read, C: Callbacks> Identity<R, C> {
|
|||
.take()
|
||||
.decrypt(self.filename.as_deref(), self.callbacks.clone())
|
||||
{
|
||||
Ok((identities, _)) => {
|
||||
let recipients = identities
|
||||
.iter()
|
||||
.map(|entry| entry.to_recipient(self.callbacks.clone()))
|
||||
.collect::<Result<Vec<_>, _>>();
|
||||
|
||||
self.state.set(IdentityState::Decrypted(identities));
|
||||
Ok((identity_file, _)) => {
|
||||
let recipients = identity_file.to_recipients(self.callbacks.clone());
|
||||
self.state.set(IdentityState::Decrypted(identity_file));
|
||||
recipients
|
||||
}
|
||||
Err(e) => {
|
||||
|
@ -158,10 +152,9 @@ impl<R: io::Read, C: Callbacks> Identity<R, C> {
|
|||
.take()
|
||||
.decrypt(self.filename.as_deref(), self.callbacks.clone())
|
||||
{
|
||||
Ok((identities, requested_passphrase)) => {
|
||||
let result = identities
|
||||
.iter()
|
||||
.map(|entry| entry.clone().into_identity(self.callbacks.clone()))
|
||||
Ok((identity_file, requested_passphrase)) => {
|
||||
let result = identity_file
|
||||
.to_identities(self.callbacks.clone())
|
||||
.find_map(filter);
|
||||
|
||||
// If we requested a passphrase to decrypt, and none of the identities
|
||||
|
@ -173,7 +166,7 @@ impl<R: io::Read, C: Callbacks> Identity<R, C> {
|
|||
));
|
||||
}
|
||||
|
||||
self.state.set(IdentityState::Decrypted(identities));
|
||||
self.state.set(IdentityState::Decrypted(identity_file));
|
||||
result
|
||||
}
|
||||
Err(e) => {
|
||||
|
|
|
@ -38,23 +38,6 @@ impl IdentityFileEntry {
|
|||
)),
|
||||
}
|
||||
}
|
||||
|
||||
#[allow(unused_variables)]
|
||||
pub(crate) fn to_recipient(
|
||||
&self,
|
||||
callbacks: impl Callbacks,
|
||||
) -> Result<Box<dyn crate::Recipient + Send>, EncryptError> {
|
||||
match self {
|
||||
IdentityFileEntry::Native(i) => Ok(Box::new(i.to_public())),
|
||||
#[cfg(feature = "plugin")]
|
||||
IdentityFileEntry::Plugin(i) => Ok(Box::new(crate::plugin::RecipientPluginV1::new(
|
||||
i.plugin(),
|
||||
&[],
|
||||
&[i.clone()],
|
||||
callbacks,
|
||||
)?)),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// A list of identities that has been parsed from some input file.
|
||||
|
@ -151,6 +134,16 @@ impl IdentityFile {
|
|||
)
|
||||
}
|
||||
|
||||
/// Returns the identities in this file.
|
||||
pub(crate) fn to_identities(
|
||||
&self,
|
||||
callbacks: impl Callbacks,
|
||||
) -> impl Iterator<Item = Result<Box<dyn crate::Identity>, DecryptError>> + '_ {
|
||||
self.identities
|
||||
.iter()
|
||||
.map(|entry| entry.clone().into_identity(callbacks.clone()))
|
||||
}
|
||||
|
||||
/// Returns the identities in this file.
|
||||
pub fn into_identities(self) -> Vec<IdentityFileEntry> {
|
||||
self.identities
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue