mirror of
https://github.com/str4d/rage.git
synced 2025-04-03 02:47:42 +03:00
Plugin docs and changelog entries
This commit is contained in:
parent
f4bee7af74
commit
d7a91adc87
6 changed files with 192 additions and 4 deletions
13
age-plugin/CHANGELOG.md
Normal file
13
age-plugin/CHANGELOG.md
Normal file
|
@ -0,0 +1,13 @@
|
|||
# Changelog
|
||||
All notable changes to the age crate will be documented in this file. Changes
|
||||
to the [age-core crate](../age-core/CHANGELOG.md) also apply to the age-plugin
|
||||
crate, and are not duplicated here.
|
||||
|
||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||
and this project adheres to Rust's notion of
|
||||
[Semantic Versioning](https://semver.org/spec/v2.0.0.html). All versions prior
|
||||
to 1.0.0 are beta releases.
|
||||
|
||||
## [Unreleased]
|
||||
|
||||
Initial beta release!
|
|
@ -2,9 +2,151 @@
|
|||
|
||||
This crate provides an API for building age plugins.
|
||||
|
||||
Currently in unstable alpha; the plugin architecture has been designed, but
|
||||
expect APIs to be altered and bugs to be fixed as we iterate. Try using it and
|
||||
[let me know how it goes!](https://github.com/str4d/rage/issues/new/choose)
|
||||
## Introduction
|
||||
|
||||
The [age file encryption format] follows the "one well-oiled joint" design philosophy.
|
||||
The mechanism for extensibility (within a particular format version) is the recipient
|
||||
stanzas within the age header: file keys can be wrapped in any number of ways, and age
|
||||
clients are required to ignore stanzas that they do not understand.
|
||||
|
||||
The core APIs that exercise this mechanism are:
|
||||
- A recipient that wraps a file key and returns a stanza.
|
||||
- An identity that unwraps a stanza and returns a file key.
|
||||
|
||||
The age plugin system provides a mechanism for exposing these core APIs across process
|
||||
boundaries. It has two main components:
|
||||
|
||||
- A map from recipients and identities to plugin binaries.
|
||||
- State machines for wrapping and unwrapping file keys.
|
||||
|
||||
With this composable design, you can implement a recipient or identity that you might
|
||||
use directly with the [`age`] library crate, and also deploy it as a plugin binary for
|
||||
use with clients like [`rage`].
|
||||
|
||||
[age file encryption format]: https://age-encryption.org/v1
|
||||
[`age`]: https://crates.io/crates/age
|
||||
[`rage`]: https://crates.io/crates/rage
|
||||
|
||||
## Mapping recipients and identities to plugin binaries
|
||||
|
||||
age plugins are identified by an arbitrary case-insensitive string `NAME`. This string
|
||||
is used in three places:
|
||||
|
||||
- Plugin-compatible recipients are encoded using Bech32 with the HRP `age1name`
|
||||
(lowercase).
|
||||
- Plugin-compatible identities are encoded using Bech32 with the HRP
|
||||
`AGE-PLUGIN-NAME-` (uppercase).
|
||||
- Plugin binaries (to be started by age clients) are named `age-plugin-name`.
|
||||
|
||||
Users interact with age clients by providing either recipients for file encryption, or
|
||||
identities for file decryption. When a plugin recipient or identity is provided, the
|
||||
age client searches the `PATH` for a binary with the corresponding plugin name.
|
||||
|
||||
Recipient stanza types are not required to be correlated to specific plugin names.
|
||||
When decrypting, age clients will pass all recipient stanzas to every connected
|
||||
plugin. Plugins MUST ignore stanzas that they do not know about.
|
||||
|
||||
A plugin binary may handle multiple recipient or identity types by being present in
|
||||
the `PATH` under multiple names. This can be implemented with symlinks or aliases to
|
||||
the canonical binary.
|
||||
|
||||
Multiple plugin binaries can support the same recipient and identity types; the first
|
||||
binary found in the `PATH` will be used by age clients. Some Unix OSs support
|
||||
"alternatives", which plugin binaries should leverage if they provide support for a
|
||||
common recipient or identity type.
|
||||
|
||||
Note that the identity specified by a user doesn't need to point to a specific
|
||||
decryption key, or indeed contain any key material at all. It only needs to contain
|
||||
sufficient information for the plugin to locate the necessary key material.
|
||||
|
||||
### Standard age keys
|
||||
|
||||
A plugin MAY support decrypting files encrypted to native age recipients, by including
|
||||
support for the `x25519` recipient stanza. Such plugins will pick their own name, and
|
||||
users will use identity files containing identities that specify that plugin name.
|
||||
|
||||
## Example plugin binary
|
||||
|
||||
The following example uses `gumdrop` to parse CLI arguments, but any argument parsing
|
||||
logic will work as long as it can detect the `--age-plugin=STATE_MACHINE` flag.
|
||||
|
||||
```rust
|
||||
use age_core::format::{FileKey, Stanza};
|
||||
use age_plugin::{
|
||||
identity::{self, Callbacks, IdentityPluginV1},
|
||||
print_new_identity,
|
||||
recipient::{self, RecipientPluginV1},
|
||||
run_state_machine,
|
||||
};
|
||||
use gumdrop::Options;
|
||||
use std::collections::HashMap;
|
||||
use std::io;
|
||||
|
||||
struct RecipientPlugin;
|
||||
|
||||
impl RecipientPluginV1 for RecipientPlugin {
|
||||
fn add_recipients<'a, I: Iterator<Item = &'a str>>(
|
||||
&mut self,
|
||||
recipients: I,
|
||||
) -> Result<(), Vec<recipient::Error>> {
|
||||
todo!()
|
||||
}
|
||||
|
||||
fn wrap_file_key(
|
||||
&mut self,
|
||||
file_key: &FileKey,
|
||||
) -> Result<Vec<Stanza>, Vec<recipient::Error>> {
|
||||
todo!()
|
||||
}
|
||||
}
|
||||
|
||||
struct IdentityPlugin;
|
||||
|
||||
impl IdentityPluginV1 for IdentityPlugin {
|
||||
fn add_identities<'a, I: Iterator<Item = &'a str>>(
|
||||
&mut self,
|
||||
identities: I,
|
||||
) -> Result<(), Vec<identity::Error>> {
|
||||
todo!()
|
||||
}
|
||||
|
||||
fn unwrap_file_keys(
|
||||
&mut self,
|
||||
files: Vec<Vec<Stanza>>,
|
||||
mut callbacks: impl Callbacks,
|
||||
) -> io::Result<HashMap<usize, Result<FileKey, Vec<identity::Error>>>> {
|
||||
todo!()
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug, Options)]
|
||||
struct PluginOptions {
|
||||
#[options(help = "print help message")]
|
||||
help: bool,
|
||||
|
||||
#[options(help = "run the given age plugin state machine", no_short)]
|
||||
age_plugin: Option<String>,
|
||||
}
|
||||
|
||||
fn main() -> io::Result<()> {
|
||||
let opts = PluginOptions::parse_args_default_or_exit();
|
||||
|
||||
if let Some(state_machine) = opts.age_plugin {
|
||||
// The plugin was started by an age client; run the state machine.
|
||||
run_state_machine(
|
||||
&state_machine,
|
||||
|| RecipientPlugin,
|
||||
|| IdentityPlugin,
|
||||
)?;
|
||||
return Ok(());
|
||||
}
|
||||
|
||||
// Here you can assume the binary is being run directly by a user,
|
||||
// and perform administrative tasks like generating keys.
|
||||
|
||||
Ok(())
|
||||
}
|
||||
```
|
||||
|
||||
## License
|
||||
|
||||
|
|
20
age-plugin/README.tpl
Normal file
20
age-plugin/README.tpl
Normal file
|
@ -0,0 +1,20 @@
|
|||
# age-plugin Rust library
|
||||
|
||||
{{readme}}
|
||||
|
||||
## License
|
||||
|
||||
Licensed under either of
|
||||
|
||||
* Apache License, Version 2.0, ([LICENSE-APACHE](../LICENSE-APACHE) or
|
||||
http://www.apache.org/licenses/LICENSE-2.0)
|
||||
* MIT license ([LICENSE-MIT](../LICENSE-MIT) or http://opensource.org/licenses/MIT)
|
||||
|
||||
at your option.
|
||||
|
||||
### Contribution
|
||||
|
||||
Unless you explicitly state otherwise, any contribution intentionally
|
||||
submitted for inclusion in the work by you, as defined in the Apache-2.0
|
||||
license, shall be dual licensed as above, without any additional terms or
|
||||
conditions.
|
|
@ -1,4 +1,4 @@
|
|||
//! This is a helper crate for implementing age plugins.
|
||||
//! This crate provides an API for building age plugins.
|
||||
//!
|
||||
//! # Introduction
|
||||
//!
|
||||
|
|
|
@ -9,6 +9,15 @@ and this project adheres to Rust's notion of
|
|||
to 1.0.0 are beta releases.
|
||||
|
||||
## [Unreleased]
|
||||
### 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 implements `age::Recipient` and runs
|
||||
the V1 recipient plugin protocol.
|
||||
- `age::plugin::IdentityPluginV1`, which implements `age::Identity` and runs
|
||||
the V1 identity plugin protocol.
|
||||
|
||||
### Changed
|
||||
- Files encrypted with this version of `age` might not decrypt with previous
|
||||
beta versions, due to changes in how stanza bodies are canonically encoded.
|
||||
|
|
|
@ -10,6 +10,10 @@ to 1.0.0 are beta releases.
|
|||
|
||||
## [Unreleased]
|
||||
### Added
|
||||
- Plugin support!
|
||||
- The new [`age-plugin`](https://crates.io/crates/age-plugin) crate provides
|
||||
a Rust API for building age plugins.
|
||||
- See https://hackmd.io/@str4d/age-plugin-spec for the beta specification.
|
||||
- The `-R/--recipients-file` flag, which accepts a path to a file containing age
|
||||
recipients, one per line (ignoring "#" prefixed comments and empty lines).
|
||||
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue