mirror of
https://github.com/str4d/rage.git
synced 2025-04-05 20:07:42 +03:00
plugins: Add request-public command to bi-directional phase
This enables plugins to request a non-secret value, which won't trigger any passphrase-specific user prompt (that e.g. hides the user's input).
This commit is contained in:
parent
933dfe1157
commit
cb443e55e0
4 changed files with 68 additions and 2 deletions
|
@ -65,6 +65,19 @@ impl<'a, 'b, R: io::Read, W: io::Write> Callbacks<Error> for BidirCallbacks<'a,
|
||||||
.map(|res| res.map(|_| ()))
|
.map(|res| res.map(|_| ()))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn request_public(&mut self, message: &str) -> plugin::Result<String, ()> {
|
||||||
|
self.0
|
||||||
|
.send("request-public", &[], message.as_bytes())
|
||||||
|
.and_then(|res| match res {
|
||||||
|
Ok(s) => String::from_utf8(s.body)
|
||||||
|
.map_err(|_| {
|
||||||
|
io::Error::new(io::ErrorKind::InvalidData, "response is not UTF-8")
|
||||||
|
})
|
||||||
|
.map(Ok),
|
||||||
|
Err(()) => Ok(Err(())),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
/// Requests a secret value from the user, such as a passphrase.
|
/// Requests a secret value from the user, such as a passphrase.
|
||||||
///
|
///
|
||||||
/// `message` will be displayed to the user, providing context for the request.
|
/// `message` will be displayed to the user, providing context for the request.
|
||||||
|
|
|
@ -227,6 +227,13 @@ pub trait Callbacks<E> {
|
||||||
/// inserting a hardware key.
|
/// inserting a hardware key.
|
||||||
fn message(&mut self, message: &str) -> age_core::plugin::Result<(), ()>;
|
fn message(&mut self, message: &str) -> age_core::plugin::Result<(), ()>;
|
||||||
|
|
||||||
|
/// Requests a non-secret value from the user.
|
||||||
|
///
|
||||||
|
/// `message` will be displayed to the user, providing context for the request.
|
||||||
|
///
|
||||||
|
/// To request secrets, use [`Callbacks::request_secret`].
|
||||||
|
fn request_public(&mut self, message: &str) -> age_core::plugin::Result<String, ()>;
|
||||||
|
|
||||||
/// Requests a secret value from the user, such as a passphrase.
|
/// Requests a secret value from the user, such as a passphrase.
|
||||||
///
|
///
|
||||||
/// `message` will be displayed to the user, providing context for the request.
|
/// `message` will be displayed to the user, providing context for the request.
|
||||||
|
|
|
@ -68,6 +68,19 @@ impl<'a, 'b, R: io::Read, W: io::Write> Callbacks<Error> for BidirCallbacks<'a,
|
||||||
.map(|res| res.map(|_| ()))
|
.map(|res| res.map(|_| ()))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn request_public(&mut self, message: &str) -> plugin::Result<String, ()> {
|
||||||
|
self.0
|
||||||
|
.send("request-public", &[], message.as_bytes())
|
||||||
|
.and_then(|res| match res {
|
||||||
|
Ok(s) => String::from_utf8(s.body)
|
||||||
|
.map_err(|_| {
|
||||||
|
io::Error::new(io::ErrorKind::InvalidData, "response is not UTF-8")
|
||||||
|
})
|
||||||
|
.map(Ok),
|
||||||
|
Err(()) => Ok(Err(())),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
/// Requests a secret value from the user, such as a passphrase.
|
/// Requests a secret value from the user, such as a passphrase.
|
||||||
///
|
///
|
||||||
/// `message` will be displayed to the user, providing context for the request.
|
/// `message` will be displayed to the user, providing context for the request.
|
||||||
|
|
|
@ -25,6 +25,7 @@ const PLUGIN_IDENTITY_PREFIX: &str = "age-plugin-";
|
||||||
const CMD_ERROR: &str = "error";
|
const CMD_ERROR: &str = "error";
|
||||||
const CMD_RECIPIENT_STANZA: &str = "recipient-stanza";
|
const CMD_RECIPIENT_STANZA: &str = "recipient-stanza";
|
||||||
const CMD_MSG: &str = "msg";
|
const CMD_MSG: &str = "msg";
|
||||||
|
const CMD_REQUEST_PUBLIC: &str = "request-public";
|
||||||
const CMD_REQUEST_SECRET: &str = "request-secret";
|
const CMD_REQUEST_SECRET: &str = "request-secret";
|
||||||
const CMD_FILE_KEY: &str = "file-key";
|
const CMD_FILE_KEY: &str = "file-key";
|
||||||
|
|
||||||
|
@ -216,13 +217,29 @@ impl<C: Callbacks> crate::Recipient for RecipientPluginV1<C> {
|
||||||
let mut stanzas = vec![];
|
let mut stanzas = vec![];
|
||||||
let mut errors = vec![];
|
let mut errors = vec![];
|
||||||
if let Err(e) = conn.bidir_receive(
|
if let Err(e) = conn.bidir_receive(
|
||||||
&[CMD_MSG, CMD_REQUEST_SECRET, CMD_RECIPIENT_STANZA, CMD_ERROR],
|
&[
|
||||||
|
CMD_MSG,
|
||||||
|
CMD_REQUEST_PUBLIC,
|
||||||
|
CMD_REQUEST_SECRET,
|
||||||
|
CMD_RECIPIENT_STANZA,
|
||||||
|
CMD_ERROR,
|
||||||
|
],
|
||||||
|mut command, reply| match command.tag.as_str() {
|
|mut command, reply| match command.tag.as_str() {
|
||||||
CMD_MSG => {
|
CMD_MSG => {
|
||||||
self.callbacks
|
self.callbacks
|
||||||
.prompt(&String::from_utf8_lossy(&command.body));
|
.prompt(&String::from_utf8_lossy(&command.body));
|
||||||
reply.ok(None)
|
reply.ok(None)
|
||||||
}
|
}
|
||||||
|
CMD_REQUEST_PUBLIC => {
|
||||||
|
if let Some(value) = self
|
||||||
|
.callbacks
|
||||||
|
.request_public_string(&String::from_utf8_lossy(&command.body))
|
||||||
|
{
|
||||||
|
reply.ok(Some(value.as_bytes()))
|
||||||
|
} else {
|
||||||
|
reply.fail()
|
||||||
|
}
|
||||||
|
}
|
||||||
CMD_REQUEST_SECRET => {
|
CMD_REQUEST_SECRET => {
|
||||||
if let Some(secret) = self
|
if let Some(secret) = self
|
||||||
.callbacks
|
.callbacks
|
||||||
|
@ -365,13 +382,29 @@ impl<C: Callbacks> IdentityPluginV1<C> {
|
||||||
let mut file_key = None;
|
let mut file_key = None;
|
||||||
let mut errors = vec![];
|
let mut errors = vec![];
|
||||||
if let Err(e) = conn.bidir_receive(
|
if let Err(e) = conn.bidir_receive(
|
||||||
&[CMD_MSG, CMD_REQUEST_SECRET, CMD_FILE_KEY, CMD_ERROR],
|
&[
|
||||||
|
CMD_MSG,
|
||||||
|
CMD_REQUEST_PUBLIC,
|
||||||
|
CMD_REQUEST_SECRET,
|
||||||
|
CMD_FILE_KEY,
|
||||||
|
CMD_ERROR,
|
||||||
|
],
|
||||||
|command, reply| match command.tag.as_str() {
|
|command, reply| match command.tag.as_str() {
|
||||||
CMD_MSG => {
|
CMD_MSG => {
|
||||||
self.callbacks
|
self.callbacks
|
||||||
.prompt(&String::from_utf8_lossy(&command.body));
|
.prompt(&String::from_utf8_lossy(&command.body));
|
||||||
reply.ok(None)
|
reply.ok(None)
|
||||||
}
|
}
|
||||||
|
CMD_REQUEST_PUBLIC => {
|
||||||
|
if let Some(value) = self
|
||||||
|
.callbacks
|
||||||
|
.request_public_string(&String::from_utf8_lossy(&command.body))
|
||||||
|
{
|
||||||
|
reply.ok(Some(value.as_bytes()))
|
||||||
|
} else {
|
||||||
|
reply.fail()
|
||||||
|
}
|
||||||
|
}
|
||||||
CMD_REQUEST_SECRET => {
|
CMD_REQUEST_SECRET => {
|
||||||
if let Some(secret) = self
|
if let Some(secret) = self
|
||||||
.callbacks
|
.callbacks
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue