mirror of
https://github.com/str4d/rage.git
synced 2025-04-03 19:07:42 +03:00
15 KiB
15 KiB
Changelog
All notable changes to the age crate will be documented in this file. Changes to the age-core crate also apply to the age crate, and are not duplicated here.
The format is based on Keep a Changelog, and this project adheres to Rust's notion of Semantic Versioning. All versions prior to 1.0.0 are beta releases.
[Unreleased]
[0.6.1] - 2024-11-18
Security
- The age plugin protocol previously allowed plugin names that could be
interpreted as file paths. Under certain conditions, this could lead to a
different binary being executed as an age plugin than intended. Plugin names
are now required to only contain alphanumeric characters or the four special
characters
+-._
.
[0.7.1] - 2021-12-27
Fixed
- Bumped
age-core
to 0.7.1 to fix a bug where non-canonical recipient stanza bodies in an age file header would cause a panic instead of being rejected.
[0.7.0] - 2021-10-18
Added
age::encrypted::Identity
, for decrypting files with passphrase-encrypted age identity files.age::IdentityFileEntry
enum, representing the possible kinds of entries within an age identity file.age::{DecryptError, EncryptError, PluginError}: Clone
bounds.age::cli_common::UiCallbacks: Clone + Copy
bounds.age::cli_common::Passphrase::random
, for generating a secure passphrase.age::cli_common::ReadError
age::secrecy
, which re-exports thesecrecy
crate.
Changed
- MSRV is now 1.51.0.
age::IdentityFile::into_identities
now returnsVec<IdentityFileEntry>
.age::cli_common::read_identities
:- Encrypted age files will now be parsed and assumed to be encrypted age identities. This assumption is checked at file-decryption time.
- New
max_work_factor
parameter for controlling the work factor when decrypting encrypted identities. - Identities are now returned in the same order as
filenames
(and top-to-bottom from within each file). Plugin identities are no longer coalesced; there is oneBox<dyn Identity>
per plugin identity. age::cli_common::ReadError
is now returned instead of a user-specified error type. The error constructor parameters have been removed from the function.
age::Callbacks::prompt
has been renamed toCallbacks::display_message
.age::cli_common::UiCallbacks::display_message
no longer usespinentry
(which displays a temporary prompt that can be dismissed), so the message is now part of the visible CLI output.
Removed
IdentityFile::split_into
(replaced byIdentityFileEntry::Plugin
).
[0.6.0] - 2021-05-02
Security
StreamReader::seek(SeekFrom::End(offset))
did not previously authenticate the ciphertext length; if the ciphertext had been truncated or extended byadversary_offset
, it would instead seek tooffset + adversary_offset
. This allowed an adversary with temporary control of an encrypted age file to control the location of a plaintext read following a seek-from-end.age
now returns an error if the last chunk is invalid.rage
was not affected by this security issue, as it does not useSeek
.rage-mount
may have been affected; it does not useSeekFrom::End
directly, but thetar
orzip
crates might do so.
Added
- Plugin support, enabled by the
plugin
feature flag:age::plugin::{Identity, Recipient}
structs for parsing plugin recipients and identities from strings.age::plugin::RecipientPluginV1
, which implementsage::Recipient
and runs the V1 recipient plugin protocol.age::plugin::IdentityPluginV1
, which implementsage::Identity
and runs the V1 identity plugin protocol.
- The
web-sys
feature flag, which enables calculating the work factor for passphrase encryption with the Performance timer via theweb-sys
crate, when compiling for a WebAssembly target such aswasm32-unknown-unknown
. This feature is ignored for thewasm32-wasi
target, which supportsstd::time::SystemTime
. age::Callbacks::request_public_string
to request non-private input from the user (which will not trigger any OS-level passphrase-style prompt, unlikeCallbacks::request_passphrase
).
Changed
- MSRV is now 1.47.0.
age::cli_common::file_io::OutputWriter::File
will now overwrite the file if it exists, instead of returning an error. This makes it consistent withage::cli_common::file_io::OutputWriter::Stdout
, as well as most UNIX tools.- Files encrypted with this version of
age
might not decrypt with previous beta versions, due to changes in how stanza bodies are canonically encoded. This should only affect a small fraction of files (if grease that triggers the change is added, which has a 3% chance per file). age::decryptor::RecipientsDecryptor
now takesimpl Iterator<Item = &'a dyn Identity>
in its decryption methods, to make decrypting multiple files with the same identities easier.age::cli_common::file_io::OutputWriter::File
now wraps aLazyFile
struct (instead of wrappingstd::io::File
directly), which does not open the file until it is first written to.age::decryptor::Callbacks
has been moved toage::Callbacks
, as it is no longer decryption-specific.
Fixed
age::cli_common::read_identities
now allows either kind of line ending in SSH identity files.- Default
en-US
language strings are now always loaded, even if translations are not loaded by callingage::localizer().select(&requested_languages)
. StreamReader::seek(SeekFrom::End(0))
now seeks to the correct position when the plaintext is an exact multiple of the chunk size.
[0.5.1] - 2021-02-13
Fixed
- Bumped dependencies to
i18n-embed-fl 0.3
andi18n-embed 0.10.2
to fix a transient dependency breakage, that brokecargo install rage
becausecargo install
ignoresCargo.lock
.
[0.5.0] - 2020-11-22
Added
- Italian, Spanish, and Chinese translations!
- New core traits, implemented by all relevant
age
types:age::Identity
, representing an identity that can decrypt an age file.age::Recipient
, representing a potential recipient of an age file.
- Separate modules and structs for different recipient types:
age::x25519
age::ssh
(behind thessh
feature flag).
age::EncryptError
, representing errors that can occur during encryption.age::IdentityFile
struct, for parsing a list of native age identities (currently onlyage::x25519::Identity
) from a file.- Asynchronous APIs for encryption and decryption, enabled by the
async
feature flag:age::Encryptor::wrap_async_output()
age::Decryptor::new_async()
age::decryptor::RecipientsDecryptor::decrypt_async()
age::decryptor::PassphraseDecryptor::decrypt_async()
- Explicit armoring support, enabled by the
armor
feature flag:age::armor::ArmoredReader
, which can be wrapped around an input to handle a potentially-armored age file.age::armor::ArmoredWriter
, which can be wrapped around an output to optionally apply the armored age format.
Changed
- MSRV is now 1.45.0.
- Changes due to the new core traits:
age::Encryptor::with_recipients
now takesVec<Box<dyn Recipient>>
.age::decryptor::RecipientsDecryptor
now takesimpl Iterator<Item = Box<dyn Identity>>
in its decryption methods.age::cli_common::read_identities
now returnsVec<Box<dyn Identity>>
, as it abstracts overage::IdentityFile
andage::ssh::Identity
. When thessh
feature flag is enabled, it also takes anunsupported_ssh
argument for handling unsupported SSH identities.age::Error
has been renamed toage::DecryptError
.
- Changes due to explicit armoring support:
age::Encryptor::wrap_output
now only generates the non-malleable binary age format. To optionally generate armored age files, useencryptor.wrap_output(ArmoredWriter::wrap_output(output, format))
.age::Decryptor
now only decrypts the non-malleable binary age format. To handle age files that are potentially armored, useDecryptor::new(ArmoredReader::new(input))
.age::Format
has been moved toage::armor::Format
.
- SSH support is now disabled by default, behind the
ssh
feature flag.ssh-rsa
keys are now supported without theunstable
feature flag. age::Callbacks
has been moved toage::decryptor::Callbacks
.
Removed
age::SecretKey
(replaced byage::x25519::Identity
andage::ssh::Identity
).age::keys::RecipientKey
(replaced byage::x25519::Recipient
andage::ssh::Recipient
).age::keys::{Identity, IdentityKey}
(replaced byage::Identity
trait on individual identities, andage::IdentityFile
for parsing identities).age::decryptor::RecipientsDecryptor::decrypt_with_callbacks()
(identities are now expected to handle their own callbacks, andage::cli_common::read_identities
now adds callbacks to SSH identities).- Default identity path:
age::cli_common::get_config_dir
.- The
no_default
parameter forage::cli_common::read_identities
.
[0.4.0] - 2020-03-25
Added
age::Decryptor::new(R: Read)
, which parses an age file header and returns a context-specific decryptor.age::decryptor
module containing the context-specific decryptors.- Their decryption methods return the concrete type
StreamReader<R>
, enabling them to handle seekable readers.
- Their decryption methods return the concrete type
age::Encryptor::with_recipients(Vec<RecipientKey>)
age::Encryptor::with_user_passphrase(SecretString)
- Support for encrypted OpenSSH keys created with
ssh-keygen
prior to OpenSSH 7.6. age::cli_common::file_io::OutputWriter::is_terminal
Changed
age::Decryptor
has been refactored to auto-detect the decryption type. As a result, both identity-based and passphrase-based decryption need to be handled.age::StreamReader
has been moved into theage::stream
module, along withStreamWriter
which was previously public but has now been formally exposed in the API for documentation purposes.age::Encryptor
is now an opaque struct, and must be created via its new constructors.age::Encryptor::wrap_output
now consumesself
, making it harder to accidentally reuse a passphrase for multiple encrypted files.age::cli_common::read_identities
now takes an additionalfile_not_found
parameter for customising the error when an identity filename is not found.
Removed
age::Decryptor::trial_decrypt
(replaced by context-specific decryptors).age::Decryptor::trial_decrypt_seekable
(merged into the context-specific decryptors).age::Error::ArmoredWhenSeeking
age::Error::MessageRequiresKeys
age::Error::MessageRequiresPassphrase
Fixed
- Key files with Windows line endings are now correctly parsed.
[0.3.1] - 2020-02-11
Fixed
- Bumped dependencies to
cookie-factory ^0.3.1
to fix nightly builds.
[0.3.0] - 2020-02-09
Added
age::Callbacks
, which encapsulates any requests that might be necessary during the decryption process.age::cli_common::UiCallbacks
, which implementsCallbacks
with requests to the user viaage::cli_common::read_secret
.age::Decryptor::with_identities(Vec<Identity>)
age::Decryptor::with_identities_and_callbacks(Vec<Identity>, Box<dyn Callbacks>)
age::Encryptor
will insert a random recipient stanza into the header, to keep age's joint well oiled.
Changed
- The CLI tools have been moved into the
rage
crate. - The
age::Decryptor::Keys
enum case has been renamed toIdentities
and altered to store aBox<dyn Callbacks>
internally. age::Decryptor::trial_decrypt
andage::Decryptor::trial_decrypt_seekable
both no longer take arequest_passphrase
argument.age::cli_common::read_secret
:- Takes an additional
prompt
parameter. - Uses the system
pinentry
binary for requesting secrets if available. - Returns
pinentry::Error
instead ofio::Error
.
- Takes an additional
age::cli_common::read_or_generate_passphrase
now returnspinentry::Error
instead ofio::Error
.- Core age parsers and serializers have been moved into the
age-core
crate.
Fixed
- Fixed several crashes in the armored format reader, found by fuzzing. The reader also now correctly enforces a canonical armor marker and line lengths.
- Recipient stanzas with empty bodies are correctly parsed.
[0.2.0] - 2020-01-10
Added
- The library crate can be compiled to WASM.
- When encrypting to a passphrase, rage will generate a secure passphrase if the user does not provide one.
SecretKey::to_string -> secrecy::SecretString
, which zeroizes most internal state. (Zeroizing all internal state requires changes to thebech32
crate.)RecipientKey
implementsDisplay
, and can be converted to a string usingrecipient.to_string()
.Decryptor::with_passphrase
constructor.--max-work-factor WF
argument for rage and rage-mount, to enable overriding the default maximum (which is around 16 seconds of work).
Changed
age::Encryptor::wrap_output
now takes anage::Format
enum argument instead of a boolean flag.- Recipients are now parsed as filenames last instead of first. If a filename happens to also be a valid recipient format, the file will be ignored. This can be overridden by using an absolute file path.
- The filename
-
(hyphen) is now treated as an explicit request to read from standard input or write to standard output when used as an input or output filename. -o -
will override protections for terminals when standard output is not being piped elsewhere: output will not be truncated, and binary data will be printed directly to the terminal.- Armored encrypted output can now be printed to the terminal. Large files will
be truncated (to protect the terminal), corrupting the encryption. This can be
overriden with
-o -
. - The
Decryptor::Passphrase
enum case has been altered to store an optional maximum work factor.
Removed
SecretKey::to_str
(replaced bySecretKey::to_string
).RecipientKey::to_str
(replaced byDisplay
implementation andrecipient.to_string()
).
Fixed
- Corrected encoding of example recipients in manpages.
- Re-enabled the default identities file (#41).
- Fixed parser to reject encrypted OpenSSH keys if they contain invalid
bcrypt_pbkdf
parameters. - [Unix]
rage-keygen -o filename
now creates files with mode600
(i.e. the output file is no longer world-readable). - Unknown recipient lines are now parsed and ignored during decryption, instead of causing a hard failure.
[0.1.1] - 2019-12-29
Added
- Debian packaging support via
cargo deb
. See docs/debian.md for details.
Changed
- Moved the
num_traits
dependency behind theunstable
feature flag. - The
generate-docs
example now generates (the equivalent of )gzip -9
manpages, for ease of use in Debian packaging.
Fixed
- Decrypted chunks inside the STREAM implementation are now zeroized after use.
[0.1.0] - 2019-12-27
Initial beta release!