plugin: fix returning in-protocol errors from plugins

Fixes Foxboron/age-plugin-tpm#31
This commit is contained in:
Filippo Valsorda
2025-12-23 12:35:06 +01:00
parent 9795b63263
commit f3b008d1b8
2 changed files with 63 additions and 1 deletions

View File

@@ -14,6 +14,7 @@ import (
"os/exec"
"path/filepath"
"runtime"
"strings"
"testing"
"filippo.io/age"
@@ -34,6 +35,15 @@ func TestMain(m *testing.M) {
return testPQCRecipient{}, nil
})
os.Exit(p.Main())
case "age-plugin-error":
p, _ := New("error")
p.HandleRecipient(func(data []byte) (age.Recipient, error) {
return nil, errors.New("oh my, an error occurred")
})
p.HandleIdentity(func(data []byte) (age.Identity, error) {
return nil, errors.New("oh my, an error occurred")
})
os.Exit(p.Main())
default:
os.Exit(m.Run())
}
@@ -163,3 +173,55 @@ func TestNotFound(t *testing.T) {
t.Errorf("expected error to wrap exec.ErrNotFound, got: %v", err)
}
}
func TestPluginError(t *testing.T) {
if runtime.GOOS == "windows" {
t.Skip("Windows support is TODO")
}
temp := t.TempDir()
testOnlyPluginPath = temp
t.Cleanup(func() { testOnlyPluginPath = "" })
ex, err := os.Executable()
if err != nil {
t.Fatal(err)
}
if err := os.Link(ex, filepath.Join(temp, "age-plugin-error")); err != nil {
t.Fatal(err)
}
if err := os.Chmod(filepath.Join(temp, "age-plugin-error"), 0755); err != nil {
t.Fatal(err)
}
r := EncodeRecipient("error", nil)
testPluginRecipient, err := NewRecipient(r, &ClientUI{})
if err != nil {
t.Fatal(err)
}
if _, err := age.Encrypt(io.Discard, testPluginRecipient); err == nil {
t.Errorf("expected error from plugin")
} else if !strings.Contains(err.Error(), "oh my, an error occurred") {
t.Errorf("expected plugin error, got: %v", err)
}
buf := &bytes.Buffer{}
id, err := age.GenerateHybridIdentity()
if err != nil {
t.Fatal(err)
}
w, err := age.Encrypt(buf, id.Recipient())
if err != nil {
t.Fatal(err)
}
w.Close()
i := EncodeIdentity("error", nil)
testPluginIdentity, err := NewIdentity(i, &ClientUI{})
if err != nil {
t.Fatal(err)
}
if _, err := age.Decrypt(buf, testPluginIdentity); err == nil {
t.Errorf("expected error from plugin")
} else if !strings.Contains(err.Error(), "oh my, an error occurred") {
t.Errorf("expected plugin error, got: %v", err)
}
}

View File

@@ -654,7 +654,7 @@ func expectUnsupported(sr *format.StanzaReader) error {
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 {
if err := s.Marshal(p.stdout); err != nil {
return fmt.Errorf("failed to write error stanza: %v", err)
}
if err := expectOk(p.sr); err != nil {