mirror of
https://github.com/FiloSottile/age.git
synced 2025-12-23 05:25:14 +00:00
@@ -5,11 +5,11 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"bufio"
|
||||
"os"
|
||||
"testing"
|
||||
|
||||
"filippo.io/age"
|
||||
"filippo.io/age/plugin"
|
||||
"github.com/rogpeppe/go-internal/testscript"
|
||||
)
|
||||
|
||||
@@ -30,51 +30,31 @@ func TestMain(m *testing.M) {
|
||||
return 0
|
||||
},
|
||||
"age-plugin-test": func() (exitCode int) {
|
||||
// TODO: use plugin server package once it's available.
|
||||
switch os.Args[1] {
|
||||
case "--age-plugin=recipient-v1":
|
||||
scanner := bufio.NewScanner(os.Stdin)
|
||||
scanner.Scan() // add-recipient
|
||||
scanner.Scan() // body
|
||||
scanner.Scan() // grease
|
||||
scanner.Scan() // body
|
||||
scanner.Scan() // wrap-file-key
|
||||
scanner.Scan() // body
|
||||
fileKey := scanner.Text()
|
||||
scanner.Scan() // extension-labels
|
||||
scanner.Scan() // body
|
||||
scanner.Scan() // done
|
||||
scanner.Scan() // body
|
||||
os.Stdout.WriteString("-> recipient-stanza 0 test\n")
|
||||
os.Stdout.WriteString(fileKey + "\n")
|
||||
scanner.Scan() // ok
|
||||
scanner.Scan() // body
|
||||
os.Stdout.WriteString("-> done\n\n")
|
||||
return 0
|
||||
case "--age-plugin=identity-v1":
|
||||
scanner := bufio.NewScanner(os.Stdin)
|
||||
scanner.Scan() // add-identity
|
||||
scanner.Scan() // body
|
||||
scanner.Scan() // grease
|
||||
scanner.Scan() // body
|
||||
scanner.Scan() // recipient-stanza
|
||||
scanner.Scan() // body
|
||||
fileKey := scanner.Text()
|
||||
scanner.Scan() // done
|
||||
scanner.Scan() // body
|
||||
os.Stdout.WriteString("-> file-key 0\n")
|
||||
os.Stdout.WriteString(fileKey + "\n")
|
||||
scanner.Scan() // ok
|
||||
scanner.Scan() // body
|
||||
os.Stdout.WriteString("-> done\n\n")
|
||||
return 0
|
||||
default:
|
||||
return 1
|
||||
}
|
||||
p, _ := plugin.New("test")
|
||||
p.HandleRecipient(func(data []byte) (age.Recipient, error) {
|
||||
return testPlugin{}, nil
|
||||
})
|
||||
p.HandleIdentity(func(data []byte) (age.Identity, error) {
|
||||
return testPlugin{}, nil
|
||||
})
|
||||
return p.Main()
|
||||
},
|
||||
}))
|
||||
}
|
||||
|
||||
type testPlugin struct{}
|
||||
|
||||
func (testPlugin) Wrap(fileKey []byte) ([]*age.Stanza, error) {
|
||||
return []*age.Stanza{{Type: "test", Body: fileKey}}, nil
|
||||
}
|
||||
|
||||
func (testPlugin) Unwrap(ss []*age.Stanza) ([]byte, error) {
|
||||
if len(ss) == 1 && ss[0].Type == "test" {
|
||||
return ss[0].Body, nil
|
||||
}
|
||||
return nil, age.ErrIncorrectIdentity
|
||||
}
|
||||
|
||||
func TestScript(t *testing.T) {
|
||||
testscript.Run(t, testscript.Params{
|
||||
Dir: "testdata",
|
||||
|
||||
@@ -4,7 +4,6 @@
|
||||
// license that can be found in the LICENSE file or at
|
||||
// https://developers.google.com/open-source/licenses/bsd
|
||||
|
||||
// Package plugin implements the age plugin protocol.
|
||||
package plugin
|
||||
|
||||
import (
|
||||
@@ -53,6 +52,15 @@ func (r *Recipient) Name() string {
|
||||
return r.name
|
||||
}
|
||||
|
||||
// String returns the recipient encoding string ("age1name1...") or
|
||||
// "<identity-based recipient>" if r was created by [Identity.Recipient].
|
||||
func (r *Recipient) String() string {
|
||||
if r.identity {
|
||||
return "<identity-based recipient>"
|
||||
}
|
||||
return r.encoding
|
||||
}
|
||||
|
||||
func (r *Recipient) Wrap(fileKey []byte) (stanzas []*age.Stanza, err error) {
|
||||
stanzas, _, err = r.WrapWithLabels(fileKey)
|
||||
return
|
||||
@@ -79,7 +87,7 @@ func (r *Recipient) WrapWithLabels(fileKey []byte) (stanzas []*age.Stanza, label
|
||||
if err := writeStanza(conn, addType, r.encoding); err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
if err := writeStanza(conn, fmt.Sprintf("grease-%x", rand.Int())); err != nil {
|
||||
if _, err := writeGrease(conn); err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
if err := writeStanzaWithBody(conn, "wrap-file-key", fileKey); err != nil {
|
||||
@@ -194,6 +202,11 @@ func (i *Identity) Name() string {
|
||||
return i.name
|
||||
}
|
||||
|
||||
// String returns the identity encoding string ("AGE-PLUGIN-NAME-1...").
|
||||
func (i *Identity) String() string {
|
||||
return i.encoding
|
||||
}
|
||||
|
||||
// Recipient returns a Recipient wrapping this identity. When that Recipient is
|
||||
// used to encrypt a file key, the identity encoding is provided as-is to the
|
||||
// plugin, which is expected to support encrypting to identities.
|
||||
@@ -223,7 +236,7 @@ func (i *Identity) Unwrap(stanzas []*age.Stanza) (fileKey []byte, err error) {
|
||||
if err := writeStanza(conn, "add-identity", i.encoding); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if err := writeStanza(conn, fmt.Sprintf("grease-%x", rand.Int())); err != nil {
|
||||
if _, err := writeGrease(conn); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
for _, rs := range stanzas {
|
||||
@@ -453,3 +466,18 @@ func writeStanzaWithBody(conn io.Writer, t string, body []byte) error {
|
||||
s := &format.Stanza{Type: t, Body: body}
|
||||
return s.Marshal(conn)
|
||||
}
|
||||
|
||||
func writeGrease(conn io.Writer) (sent bool, err error) {
|
||||
if rand.Intn(3) == 0 {
|
||||
return false, nil
|
||||
}
|
||||
s := &format.Stanza{Type: fmt.Sprintf("grease-%x", rand.Int())}
|
||||
for i := 0; i < rand.Intn(3); i++ {
|
||||
s.Args = append(s.Args, fmt.Sprintf("%d", rand.Intn(100)))
|
||||
}
|
||||
if rand.Intn(2) == 0 {
|
||||
s.Body = make([]byte, rand.Intn(100))
|
||||
rand.Read(s.Body)
|
||||
}
|
||||
return true, s.Marshal(conn)
|
||||
}
|
||||
|
||||
@@ -7,7 +7,6 @@
|
||||
package plugin
|
||||
|
||||
import (
|
||||
"bufio"
|
||||
"io"
|
||||
"os"
|
||||
"path/filepath"
|
||||
@@ -20,63 +19,41 @@ import (
|
||||
|
||||
func TestMain(m *testing.M) {
|
||||
switch filepath.Base(os.Args[0]) {
|
||||
// TODO: deduplicate from cmd/age TestMain.
|
||||
case "age-plugin-test":
|
||||
switch os.Args[1] {
|
||||
case "--age-plugin=recipient-v1":
|
||||
scanner := bufio.NewScanner(os.Stdin)
|
||||
scanner.Scan() // add-recipient
|
||||
scanner.Scan() // body
|
||||
scanner.Scan() // grease
|
||||
scanner.Scan() // body
|
||||
scanner.Scan() // wrap-file-key
|
||||
scanner.Scan() // body
|
||||
fileKey := scanner.Text()
|
||||
scanner.Scan() // extension-labels
|
||||
scanner.Scan() // body
|
||||
scanner.Scan() // done
|
||||
scanner.Scan() // body
|
||||
os.Stdout.WriteString("-> recipient-stanza 0 test\n")
|
||||
os.Stdout.WriteString(fileKey + "\n")
|
||||
scanner.Scan() // ok
|
||||
scanner.Scan() // body
|
||||
os.Stdout.WriteString("-> done\n\n")
|
||||
os.Exit(0)
|
||||
default:
|
||||
panic(os.Args[1])
|
||||
}
|
||||
p, _ := New("test")
|
||||
p.HandleRecipient(func(data []byte) (age.Recipient, error) {
|
||||
return testRecipient{}, nil
|
||||
})
|
||||
os.Exit(p.Main())
|
||||
case "age-plugin-testpqc":
|
||||
switch os.Args[1] {
|
||||
case "--age-plugin=recipient-v1":
|
||||
scanner := bufio.NewScanner(os.Stdin)
|
||||
scanner.Scan() // add-recipient
|
||||
scanner.Scan() // body
|
||||
scanner.Scan() // grease
|
||||
scanner.Scan() // body
|
||||
scanner.Scan() // wrap-file-key
|
||||
scanner.Scan() // body
|
||||
fileKey := scanner.Text()
|
||||
scanner.Scan() // extension-labels
|
||||
scanner.Scan() // body
|
||||
scanner.Scan() // done
|
||||
scanner.Scan() // body
|
||||
os.Stdout.WriteString("-> recipient-stanza 0 test\n")
|
||||
os.Stdout.WriteString(fileKey + "\n")
|
||||
scanner.Scan() // ok
|
||||
scanner.Scan() // body
|
||||
os.Stdout.WriteString("-> labels postquantum\n\n")
|
||||
scanner.Scan() // ok
|
||||
scanner.Scan() // body
|
||||
os.Stdout.WriteString("-> done\n\n")
|
||||
os.Exit(0)
|
||||
default:
|
||||
panic(os.Args[1])
|
||||
}
|
||||
p, _ := New("testpqc")
|
||||
p.HandleRecipient(func(data []byte) (age.Recipient, error) {
|
||||
return testPQCRecipient{}, nil
|
||||
})
|
||||
os.Exit(p.Main())
|
||||
default:
|
||||
os.Exit(m.Run())
|
||||
}
|
||||
}
|
||||
|
||||
type testRecipient struct{}
|
||||
|
||||
func (testRecipient) Wrap(fileKey []byte) ([]*age.Stanza, error) {
|
||||
return []*age.Stanza{{Type: "test", Body: fileKey}}, nil
|
||||
}
|
||||
|
||||
type testPQCRecipient struct{}
|
||||
|
||||
var _ age.RecipientWithLabels = testPQCRecipient{}
|
||||
|
||||
func (testPQCRecipient) Wrap(fileKey []byte) ([]*age.Stanza, error) {
|
||||
return []*age.Stanza{{Type: "test", Body: fileKey}}, nil
|
||||
}
|
||||
|
||||
func (testPQCRecipient) WrapWithLabels(fileKey []byte) ([]*age.Stanza, []string, error) {
|
||||
return []*age.Stanza{{Type: "test", Body: fileKey}}, []string{"postquantum"}, nil
|
||||
}
|
||||
|
||||
func TestLabels(t *testing.T) {
|
||||
if runtime.GOOS == "windows" {
|
||||
t.Skip("Windows support is TODO")
|
||||
|
||||
43
plugin/example_test.go
Normal file
43
plugin/example_test.go
Normal file
@@ -0,0 +1,43 @@
|
||||
package plugin_test
|
||||
|
||||
import (
|
||||
"log"
|
||||
"os"
|
||||
|
||||
"filippo.io/age"
|
||||
"filippo.io/age/plugin"
|
||||
)
|
||||
|
||||
type Recipient struct{}
|
||||
|
||||
func (r *Recipient) Wrap(fileKey []byte) ([]*age.Stanza, error) {
|
||||
panic("unimplemented")
|
||||
}
|
||||
|
||||
func NewRecipient(data []byte) (*Recipient, error) {
|
||||
return &Recipient{}, nil
|
||||
}
|
||||
|
||||
type Identity struct{}
|
||||
|
||||
func (i *Identity) Unwrap(s []*age.Stanza) ([]byte, error) {
|
||||
panic("unimplemented")
|
||||
}
|
||||
|
||||
func NewIdentity(data []byte) (*Identity, error) {
|
||||
return &Identity{}, nil
|
||||
}
|
||||
|
||||
func ExamplePlugin_main() {
|
||||
p, err := plugin.New("example")
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
p.HandleRecipient(func(data []byte) (age.Recipient, error) {
|
||||
return NewRecipient(data)
|
||||
})
|
||||
p.HandleIdentity(func(data []byte) (age.Identity, error) {
|
||||
return NewIdentity(data)
|
||||
})
|
||||
os.Exit(p.Main())
|
||||
}
|
||||
676
plugin/plugin.go
Normal file
676
plugin/plugin.go
Normal file
@@ -0,0 +1,676 @@
|
||||
// Package plugin implements the age plugin protocol.
|
||||
//
|
||||
// [Recipient] and [Indentity] are plugin clients, that execute plugin binaries to
|
||||
// perform encryption and decryption operations.
|
||||
//
|
||||
// [Plugin] is a framework for writing age plugins, that exposes an [age.Recipient]
|
||||
// and/or [age.Identity] implementation as a plugin binary.
|
||||
package plugin
|
||||
|
||||
import (
|
||||
"bufio"
|
||||
"errors"
|
||||
"flag"
|
||||
"fmt"
|
||||
"io"
|
||||
"os"
|
||||
"strconv"
|
||||
|
||||
"filippo.io/age"
|
||||
"filippo.io/age/internal/format"
|
||||
)
|
||||
|
||||
// TODO: add plugin test framework.
|
||||
|
||||
// Plugin is a framework for writing age plugins. It allows exposing regular
|
||||
// [age.Recipient] and [age.Identity] implementations as plugins, and handles
|
||||
// all the protocol details.
|
||||
type Plugin struct {
|
||||
name string
|
||||
fs *flag.FlagSet
|
||||
sm *string
|
||||
|
||||
recipient func([]byte) (age.Recipient, error)
|
||||
idAsRecipient func([]byte) (age.Recipient, error)
|
||||
identity func([]byte) (age.Identity, error)
|
||||
|
||||
stdin io.Reader
|
||||
stdout, stderr io.Writer
|
||||
|
||||
sr *format.StanzaReader
|
||||
// broken is set if the protocol broke down during an interaction function
|
||||
// called by a Recipient or Identity.
|
||||
broken bool
|
||||
}
|
||||
|
||||
// New creates a new Plugin with the given name.
|
||||
//
|
||||
// For example, a plugin named "frood" would be invoked as "age-plugin-frood".
|
||||
func New(name string) (*Plugin, error) {
|
||||
return &Plugin{name: name, stdin: os.Stdin,
|
||||
stdout: os.Stdout, stderr: os.Stderr}, nil
|
||||
}
|
||||
|
||||
// Name returns the name of the plugin.
|
||||
func (p *Plugin) Name() string {
|
||||
return p.name
|
||||
}
|
||||
|
||||
// RegisterFlags registers the plugin's flags with the given [flag.FlagSet], or
|
||||
// with the default [flag.CommandLine] if fs is nil. It must be called before
|
||||
// [flag.Parse] and [Plugin.Main].
|
||||
//
|
||||
// This allows the plugin to expose additional flags when invoked manually, for
|
||||
// example to implement a keygen mode.
|
||||
func (p *Plugin) RegisterFlags(fs *flag.FlagSet) {
|
||||
if fs == nil {
|
||||
fs = flag.CommandLine
|
||||
}
|
||||
p.fs = fs
|
||||
p.sm = fs.String("age-plugin", "", "age-plugin state machine")
|
||||
}
|
||||
|
||||
// HandleRecipient registers a function to parse recipients of the form
|
||||
// age1name1... into [age.Recipient] values. data is the decoded Bech32 payload.
|
||||
//
|
||||
// If the returned Recipient implements [age.RecipientWithLabels], Plugin will
|
||||
// use it and enforce consistency across every returned stanza in an execution.
|
||||
// If the client supports labels, they will be passed through the protocol.
|
||||
//
|
||||
// It must be called before [Plugin.Main], and can be called at most once.
|
||||
// Otherwise, it panics.
|
||||
func (p *Plugin) HandleRecipient(f func(data []byte) (age.Recipient, error)) {
|
||||
if p.recipient != nil {
|
||||
panic("HandleRecipient called twice")
|
||||
}
|
||||
p.recipient = f
|
||||
}
|
||||
|
||||
// HandleIdentityAsRecipient registers a function to parse identities of the
|
||||
// form AGE-PLUGIN-NAME-1... into [age.Recipient] values, for when identities
|
||||
// are used as recipients. data is the decoded Bech32 payload.
|
||||
//
|
||||
// If the returned Recipient implements [age.RecipientWithLabels], Plugin will
|
||||
// use it and enforce consistency across every returned stanza in an execution.
|
||||
// If the client supports labels, they will be passed through the protocol.
|
||||
//
|
||||
// It must be called before [Plugin.Main], and can be called at most once.
|
||||
// Otherwise, it panics.
|
||||
func (p *Plugin) HandleIdentityAsRecipient(f func(data []byte) (age.Recipient, error)) {
|
||||
if p.idAsRecipient != nil {
|
||||
panic("HandleIdentityAsRecipient called twice")
|
||||
}
|
||||
p.idAsRecipient = f
|
||||
}
|
||||
|
||||
// HandleIdentity registers a function to parse identities of the form
|
||||
// AGE-PLUGIN-NAME-1... into [age.Identity] values. data is the decoded Bech32
|
||||
// payload.
|
||||
//
|
||||
// It must be called before [Plugin.Main], and can be called at most once.
|
||||
// Otherwise, it panics.
|
||||
func (p *Plugin) HandleIdentity(f func(data []byte) (age.Identity, error)) {
|
||||
if p.identity != nil {
|
||||
panic("HandleIdentity called twice")
|
||||
}
|
||||
p.identity = f
|
||||
}
|
||||
|
||||
// HandleRecipientEncoding is like [Plugin.HandleRecipient] but provides the
|
||||
// full recipient encoding string to the callback.
|
||||
//
|
||||
// It allows using functions like ParseRecipient directly.
|
||||
func (p *Plugin) HandleRecipientEncoding(f func(recipient string) (age.Recipient, error)) {
|
||||
p.HandleRecipient(func(data []byte) (age.Recipient, error) {
|
||||
return f(EncodeRecipient(p.name, data))
|
||||
})
|
||||
}
|
||||
|
||||
// HandleIdentityEncodingAsRecipient is like [Plugin.HandleIdentityAsRecipient] but
|
||||
// provides the full identity encoding string to the callback.
|
||||
func (p *Plugin) HandleIdentityEncodingAsRecipient(f func(identity string) (age.Recipient, error)) {
|
||||
p.HandleIdentityAsRecipient(func(data []byte) (age.Recipient, error) {
|
||||
return f(EncodeIdentity(p.name, data))
|
||||
})
|
||||
}
|
||||
|
||||
// HandleIdentityEncoding is like [Plugin.HandleIdentity] but provides the
|
||||
// full identity encoding string to the callback.
|
||||
//
|
||||
// It allows using functions like ParseIdentity directly.
|
||||
func (p *Plugin) HandleIdentityEncoding(f func(identity string) (age.Identity, error)) {
|
||||
p.HandleIdentity(func(data []byte) (age.Identity, error) {
|
||||
return f(EncodeIdentity(p.name, data))
|
||||
})
|
||||
}
|
||||
|
||||
// Main runs the plugin protocol. It returns an exit code to pass to os.Exit.
|
||||
//
|
||||
// It automatically calls [Plugin.RegisterFlags] and [flag.Parse] if they were
|
||||
// not called before.
|
||||
func (p *Plugin) Main() int {
|
||||
if p.fs == nil {
|
||||
p.RegisterFlags(nil)
|
||||
}
|
||||
if !p.fs.Parsed() {
|
||||
p.fs.Parse(os.Args[1:])
|
||||
}
|
||||
if *p.sm == "recipient-v1" {
|
||||
return p.RecipientV1()
|
||||
}
|
||||
if *p.sm == "identity-v1" {
|
||||
return p.IdentityV1()
|
||||
}
|
||||
fmt.Fprintf(p.stderr, "unknown state machine %q", *p.sm)
|
||||
return 4
|
||||
}
|
||||
|
||||
// SetIO sets the plugin's input and output streams, which default to
|
||||
// stdin/stdout/stderr.
|
||||
//
|
||||
// It must be called before [Plugin.Main].
|
||||
func (p *Plugin) SetIO(stdin io.Reader, stdout, stderr io.Writer) {
|
||||
p.stdin = stdin
|
||||
p.stdout = stdout
|
||||
p.stderr = stderr
|
||||
}
|
||||
|
||||
// RecipientV1 implements the recipient-v1 state machine. It returns an exit
|
||||
// code to pass to os.Exit.
|
||||
//
|
||||
// Most plugins should call [Plugin.Main] instead of this method.
|
||||
func (p *Plugin) RecipientV1() int {
|
||||
if p.recipient == nil && p.idAsRecipient == nil {
|
||||
return p.fatalf("recipient-v1 not supported")
|
||||
}
|
||||
|
||||
var recipientStrings, identityStrings []string
|
||||
var fileKeys [][]byte
|
||||
var supportsLabels bool
|
||||
|
||||
p.sr = format.NewStanzaReader(bufio.NewReader(p.stdin))
|
||||
ReadLoop:
|
||||
for {
|
||||
s, err := p.sr.ReadStanza()
|
||||
if err != nil {
|
||||
return p.fatalf("failed to read stanza: %v", err)
|
||||
}
|
||||
|
||||
switch s.Type {
|
||||
case "add-recipient":
|
||||
if err := expectStanzaWithNoBody(s, 1); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
recipientStrings = append(recipientStrings, s.Args[0])
|
||||
case "add-identity":
|
||||
if err := expectStanzaWithNoBody(s, 1); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
identityStrings = append(identityStrings, s.Args[0])
|
||||
case "extension-labels":
|
||||
if err := expectStanzaWithNoBody(s, 0); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
supportsLabels = true
|
||||
case "wrap-file-key":
|
||||
if err := expectStanzaWithBody(s, 0); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
fileKeys = append(fileKeys, s.Body)
|
||||
case "done":
|
||||
if err := expectStanzaWithNoBody(s, 0); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
break ReadLoop
|
||||
default:
|
||||
// Unsupported stanzas in uni-directional phases are ignored.
|
||||
}
|
||||
}
|
||||
|
||||
if len(recipientStrings)+len(identityStrings) == 0 {
|
||||
return p.fatalf("no recipients or identities provided")
|
||||
}
|
||||
if len(fileKeys) == 0 {
|
||||
return p.fatalf("no file keys provided")
|
||||
}
|
||||
|
||||
var recipients, identities []age.Recipient
|
||||
for i, s := range recipientStrings {
|
||||
name, data, err := ParseRecipient(s)
|
||||
if err != nil {
|
||||
return p.recipientError(i, err)
|
||||
}
|
||||
if name != p.name {
|
||||
return p.recipientError(i, fmt.Errorf("unsupported plugin name: %q", name))
|
||||
}
|
||||
if p.recipient == nil {
|
||||
return p.recipientError(i, fmt.Errorf("recipient encodings not supported"))
|
||||
}
|
||||
r, err := p.recipient(data)
|
||||
if err != nil {
|
||||
return p.recipientError(i, err)
|
||||
}
|
||||
recipients = append(recipients, r)
|
||||
}
|
||||
for i, s := range identityStrings {
|
||||
name, data, err := ParseIdentity(s)
|
||||
if err != nil {
|
||||
return p.identityError(i, err)
|
||||
}
|
||||
if name != p.name {
|
||||
return p.identityError(i, fmt.Errorf("unsupported plugin name: %q", name))
|
||||
}
|
||||
if p.idAsRecipient == nil {
|
||||
return p.identityError(i, fmt.Errorf("identity encodings not supported"))
|
||||
}
|
||||
r, err := p.idAsRecipient(data)
|
||||
if err != nil {
|
||||
return p.identityError(i, err)
|
||||
}
|
||||
identities = append(identities, r)
|
||||
}
|
||||
|
||||
// Technically labels should be per-file key, but the client-side protocol
|
||||
// extension shipped like this, and it doesn't feel worth making a v2.
|
||||
var labels []string
|
||||
|
||||
stanzas := make([][]*age.Stanza, len(fileKeys))
|
||||
for i, fk := range fileKeys {
|
||||
for j, r := range recipients {
|
||||
ss, ll, err := wrapWithLabels(r, fk)
|
||||
if p.broken {
|
||||
return 2
|
||||
} else if err != nil {
|
||||
return p.recipientError(j, err)
|
||||
}
|
||||
if i == 0 && j == 0 {
|
||||
labels = ll
|
||||
} else if err := checkLabels(ll, labels); err != nil {
|
||||
return p.recipientError(j, err)
|
||||
}
|
||||
stanzas[i] = append(stanzas[i], ss...)
|
||||
}
|
||||
for j, r := range identities {
|
||||
ss, ll, err := wrapWithLabels(r, fk)
|
||||
if p.broken {
|
||||
return 2
|
||||
} else if err != nil {
|
||||
return p.identityError(j, err)
|
||||
}
|
||||
if i == 0 && j == 0 && len(recipients) == 0 {
|
||||
labels = ll
|
||||
} else if err := checkLabels(ll, labels); err != nil {
|
||||
return p.identityError(j, err)
|
||||
}
|
||||
stanzas[i] = append(stanzas[i], ss...)
|
||||
}
|
||||
}
|
||||
|
||||
if sent, err := writeGrease(p.stdout); err != nil {
|
||||
return p.fatalf("failed to write grease: %v", err)
|
||||
} else if sent {
|
||||
if err := expectUnsupported(p.sr); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
}
|
||||
|
||||
if supportsLabels {
|
||||
if err := writeStanza(p.stdout, "labels", labels...); err != nil {
|
||||
return p.fatalf("failed to write labels stanza: %v", err)
|
||||
}
|
||||
if err := expectOk(p.sr); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
}
|
||||
|
||||
for i, ss := range stanzas {
|
||||
for _, s := range ss {
|
||||
if err := (&format.Stanza{Type: "recipient-stanza",
|
||||
Args: append([]string{fmt.Sprint(i), s.Type}, s.Args...),
|
||||
Body: s.Body}).Marshal(p.stdout); err != nil {
|
||||
return p.fatalf("failed to write recipient-stanza: %v", err)
|
||||
}
|
||||
if err := expectOk(p.sr); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
}
|
||||
if sent, err := writeGrease(p.stdout); err != nil {
|
||||
return p.fatalf("failed to write grease: %v", err)
|
||||
} else if sent {
|
||||
if err := expectUnsupported(p.sr); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if err := writeStanza(p.stdout, "done"); err != nil {
|
||||
return p.fatalf("failed to write done stanza: %v", err)
|
||||
}
|
||||
return 0
|
||||
}
|
||||
|
||||
func wrapWithLabels(r age.Recipient, fileKey []byte) ([]*age.Stanza, []string, error) {
|
||||
if r, ok := r.(age.RecipientWithLabels); ok {
|
||||
return r.WrapWithLabels(fileKey)
|
||||
}
|
||||
s, err := r.Wrap(fileKey)
|
||||
return s, nil, err
|
||||
}
|
||||
|
||||
func checkLabels(ll, labels []string) error {
|
||||
if !slicesEqual(ll, labels) {
|
||||
return fmt.Errorf("labels %q do not match previous recipients %q", ll, labels)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// IdentityV1 implements the identity-v1 state machine. It returns an exit code
|
||||
// to pass to os.Exit.
|
||||
//
|
||||
// Most plugins should call [Plugin.Main] instead of this method.
|
||||
func (p *Plugin) IdentityV1() int {
|
||||
if p.identity == nil {
|
||||
return p.fatalf("identity-v1 not supported")
|
||||
}
|
||||
|
||||
var files [][]*age.Stanza
|
||||
var identityStrings []string
|
||||
|
||||
p.sr = format.NewStanzaReader(bufio.NewReader(p.stdin))
|
||||
ReadLoop:
|
||||
for {
|
||||
s, err := p.sr.ReadStanza()
|
||||
if err != nil {
|
||||
return p.fatalf("failed to read stanza: %v", err)
|
||||
}
|
||||
|
||||
switch s.Type {
|
||||
case "add-identity":
|
||||
if err := expectStanzaWithNoBody(s, 1); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
identityStrings = append(identityStrings, s.Args[0])
|
||||
case "recipient-stanza":
|
||||
if len(s.Args) < 2 {
|
||||
return p.fatalf("recipient-stanza stanza has %d arguments, want >=2", len(s.Args))
|
||||
}
|
||||
i, err := strconv.Atoi(s.Args[0])
|
||||
if err != nil {
|
||||
return p.fatalf("failed to parse recipient-stanza stanza argument: %v", err)
|
||||
}
|
||||
ss := &age.Stanza{Type: s.Args[1], Args: s.Args[2:], Body: s.Body}
|
||||
switch i {
|
||||
case len(files):
|
||||
files = append(files, []*age.Stanza{ss})
|
||||
case len(files) - 1:
|
||||
files[len(files)-1] = append(files[len(files)-1], ss)
|
||||
default:
|
||||
return p.fatalf("unexpected file index %d, previous was %d", i, len(files)-1)
|
||||
}
|
||||
case "done":
|
||||
if err := expectStanzaWithNoBody(s, 0); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
break ReadLoop
|
||||
default:
|
||||
// Unsupported stanzas in uni-directional phases are ignored.
|
||||
}
|
||||
}
|
||||
|
||||
if len(identityStrings) == 0 {
|
||||
return p.fatalf("no identities provided")
|
||||
}
|
||||
if len(files) == 0 {
|
||||
return p.fatalf("no stanzas provided")
|
||||
}
|
||||
|
||||
var identities []age.Identity
|
||||
for i, s := range identityStrings {
|
||||
name, data, err := ParseIdentity(s)
|
||||
if err != nil {
|
||||
return p.identityError(i, err)
|
||||
}
|
||||
if name != p.name {
|
||||
return p.identityError(i, fmt.Errorf("unsupported plugin name: %q", name))
|
||||
}
|
||||
if p.identity == nil {
|
||||
return p.identityError(i, fmt.Errorf("identity encodings not supported"))
|
||||
}
|
||||
r, err := p.identity(data)
|
||||
if err != nil {
|
||||
return p.identityError(i, err)
|
||||
}
|
||||
identities = append(identities, r)
|
||||
}
|
||||
|
||||
for i, ss := range files {
|
||||
if sent, err := writeGrease(p.stdout); err != nil {
|
||||
return p.fatalf("failed to write grease: %v", err)
|
||||
} else if sent {
|
||||
if err := expectUnsupported(p.sr); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
}
|
||||
|
||||
// TODO: there should be a mechanism to let the plugin decide the order
|
||||
// in which identities are tried.
|
||||
for _, id := range identities {
|
||||
fk, err := id.Unwrap(ss)
|
||||
if p.broken {
|
||||
return 2
|
||||
} else if errors.Is(err, age.ErrIncorrectIdentity) {
|
||||
continue
|
||||
} else if err != nil {
|
||||
if err := p.writeError([]string{"stanza", fmt.Sprint(i), "0"}, err); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
// Note that we don't exit here, as the protocol allows
|
||||
// continuing with other files.
|
||||
break
|
||||
}
|
||||
|
||||
s := &format.Stanza{Type: "file-key", Args: []string{fmt.Sprint(i)}, Body: fk}
|
||||
if err := s.Marshal(p.stdout); err != nil {
|
||||
return p.fatalf("failed to write file-key: %v", err)
|
||||
}
|
||||
if err := expectOk(p.sr); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
if err := writeStanza(p.stdout, "done"); err != nil {
|
||||
return p.fatalf("failed to write done stanza: %v", err)
|
||||
}
|
||||
return 0
|
||||
}
|
||||
|
||||
// DisplayMessage requests that the client display a message to the user. The
|
||||
// message should start with a lowercase letter and have no final period.
|
||||
// DisplayMessage returns an error if the client can't display the message, and
|
||||
// may return before the message has been displayed to the user.
|
||||
//
|
||||
// It must only be called by a Wrap or Unwrap method invoked by [Plugin.Main].
|
||||
func (p *Plugin) DisplayMessage(message string) error {
|
||||
if err := writeStanzaWithBody(p.stdout, "msg", []byte(message)); err != nil {
|
||||
return p.fatalInteractf("failed to write msg stanza: %v", err)
|
||||
}
|
||||
s, err := readOkOrFail(p.sr)
|
||||
if err != nil {
|
||||
return p.fatalInteractf("%v", err)
|
||||
}
|
||||
if s.Type == "fail" {
|
||||
return fmt.Errorf("client failed to display message")
|
||||
}
|
||||
if err := expectStanzaWithNoBody(s, 0); err != nil {
|
||||
return p.fatalInteractf("%v", err)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// RequestValue requests a secret or public input from the user through the
|
||||
// client, with the provided prompt. It returns an error if the client can't
|
||||
// request the input or if the user dismisses the prompt.
|
||||
//
|
||||
// It must only be called by a Wrap or Unwrap method invoked by [Plugin.Main].
|
||||
func (p *Plugin) RequestValue(prompt string, secret bool) (string, error) {
|
||||
t := "request-public"
|
||||
if secret {
|
||||
t = "request-secret"
|
||||
}
|
||||
if err := writeStanzaWithBody(p.stdout, t, []byte(prompt)); err != nil {
|
||||
return "", p.fatalInteractf("failed to write stanza: %v", err)
|
||||
}
|
||||
s, err := readOkOrFail(p.sr)
|
||||
if err != nil {
|
||||
return "", p.fatalInteractf("%v", err)
|
||||
}
|
||||
if s.Type == "fail" {
|
||||
return "", fmt.Errorf("client failed to request value")
|
||||
}
|
||||
if err := expectStanzaWithBody(s, 0); err != nil {
|
||||
return "", p.fatalInteractf("%v", err)
|
||||
}
|
||||
return string(s.Body), nil
|
||||
}
|
||||
|
||||
// Confirm requests a confirmation from the user through the client, with the
|
||||
// provided prompt. The yes and no value are the choices provided to the user.
|
||||
// no may be empty. The return value choseYes indicates whether the user
|
||||
// selected the yes or no option. Confirm returns an error if the client can't
|
||||
// request the confirmation.
|
||||
//
|
||||
// It must only be called by a Wrap or Unwrap method invoked by [Plugin.Main].
|
||||
func (p *Plugin) Confirm(prompt, yes, no string) (choseYes bool, err error) {
|
||||
args := []string{format.EncodeToString([]byte(yes))}
|
||||
if no != "" {
|
||||
args = append(args, format.EncodeToString([]byte(no)))
|
||||
}
|
||||
s := &format.Stanza{Type: "confirm", Args: args, Body: []byte(prompt)}
|
||||
if err := s.Marshal(p.stdout); err != nil {
|
||||
return false, p.fatalInteractf("failed to write confirm stanza: %v", err)
|
||||
}
|
||||
s, err = readOkOrFail(p.sr)
|
||||
if err != nil {
|
||||
return false, p.fatalInteractf("%v", err)
|
||||
}
|
||||
if s.Type == "fail" {
|
||||
return false, fmt.Errorf("client failed to request confirmation")
|
||||
}
|
||||
if err := expectStanzaWithNoBody(s, 1); err != nil {
|
||||
return false, p.fatalInteractf("%v", err)
|
||||
}
|
||||
return s.Args[0] == "yes", nil
|
||||
}
|
||||
|
||||
// fatalInteractf prints the error to stderr and sets the broken flag, so the
|
||||
// Wrap/Unwrap caller can exit with an error.
|
||||
func (p *Plugin) fatalInteractf(format string, args ...interface{}) error {
|
||||
p.broken = true
|
||||
fmt.Fprintf(p.stderr, format, args...)
|
||||
return fmt.Errorf(format, args...)
|
||||
}
|
||||
|
||||
func (p *Plugin) fatalf(format string, args ...interface{}) int {
|
||||
fmt.Fprintf(p.stderr, format, args...)
|
||||
return 1
|
||||
}
|
||||
|
||||
func expectStanzaWithNoBody(s *format.Stanza, wantArgs int) error {
|
||||
if len(s.Args) != wantArgs {
|
||||
return fmt.Errorf("%s stanza has %d arguments, want %d", s.Type, len(s.Args), wantArgs)
|
||||
}
|
||||
if len(s.Body) != 0 {
|
||||
return fmt.Errorf("%s stanza has %d bytes of body, want 0", s.Type, len(s.Body))
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func expectStanzaWithBody(s *format.Stanza, wantArgs int) error {
|
||||
if len(s.Args) != wantArgs {
|
||||
return fmt.Errorf("%s stanza has %d arguments, want %d", s.Type, len(s.Args), wantArgs)
|
||||
}
|
||||
if len(s.Body) == 0 {
|
||||
return fmt.Errorf("%s stanza has 0 bytes of body, want >0", s.Type)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (p *Plugin) recipientError(idx int, err error) int {
|
||||
if err := p.writeError([]string{"recipient", fmt.Sprint(idx)}, err); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
return 3
|
||||
}
|
||||
|
||||
func (p *Plugin) identityError(idx int, err error) int {
|
||||
if err := p.writeError([]string{"identity", fmt.Sprint(idx)}, err); err != nil {
|
||||
return p.fatalf("%v", err)
|
||||
}
|
||||
return 3
|
||||
}
|
||||
|
||||
func expectOk(sr *format.StanzaReader) error {
|
||||
ok, err := sr.ReadStanza()
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to read OK stanza: %v", err)
|
||||
}
|
||||
if ok.Type != "ok" {
|
||||
return fmt.Errorf("expected OK stanza, got %q", ok.Type)
|
||||
}
|
||||
return expectStanzaWithNoBody(ok, 0)
|
||||
}
|
||||
|
||||
func readOkOrFail(sr *format.StanzaReader) (*format.Stanza, error) {
|
||||
s, err := sr.ReadStanza()
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to read response stanza: %v", err)
|
||||
}
|
||||
switch s.Type {
|
||||
case "fail":
|
||||
if err := expectStanzaWithNoBody(s, 0); err != nil {
|
||||
return nil, fmt.Errorf("%v", err)
|
||||
}
|
||||
return s, nil
|
||||
case "ok":
|
||||
return s, nil
|
||||
default:
|
||||
return nil, fmt.Errorf("expected ok or fail stanza, got %q", s.Type)
|
||||
}
|
||||
}
|
||||
|
||||
func expectUnsupported(sr *format.StanzaReader) error {
|
||||
unsupported, err := sr.ReadStanza()
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to read unsupported stanza: %v", err)
|
||||
}
|
||||
if unsupported.Type != "unsupported" {
|
||||
return fmt.Errorf("expected unsupported stanza, got %q", unsupported.Type)
|
||||
}
|
||||
return expectStanzaWithNoBody(unsupported, 0)
|
||||
}
|
||||
|
||||
func (p *Plugin) writeError(args []string, err error) error {
|
||||
s := &format.Stanza{Type: "error", Args: args}
|
||||
s.Body = []byte(err.Error())
|
||||
if err := s.Marshal(p.stderr); err != nil {
|
||||
return fmt.Errorf("failed to write error stanza: %v", err)
|
||||
}
|
||||
if err := expectOk(p.sr); err != nil {
|
||||
return fmt.Errorf("%v", err)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func slicesEqual(s1, s2 []string) bool {
|
||||
if len(s1) != len(s2) {
|
||||
return false
|
||||
}
|
||||
for i := range s1 {
|
||||
if s1[i] != s2[i] {
|
||||
return false
|
||||
}
|
||||
}
|
||||
return true
|
||||
}
|
||||
Reference in New Issue
Block a user