Add :save and :pipe commands to viewer

* :save takes a path and saves the current message part to that location
* :pipe is the same as pipe on the account page, but uses the current
  message part rather than the whole email (ie :pipe gzip -d)
* Refactored account:pipe and extracted common pipe code to
  commands.util.QuickTerm
* Added helper command aerc.PushError
This commit is contained in:
Galen Abell 2019-05-26 17:37:39 -04:00 committed by Drew DeVault
parent 62cd0b08aa
commit 28fc9fa53d
8 changed files with 194 additions and 37 deletions

View file

@ -3,12 +3,9 @@ package account
import ( import (
"errors" "errors"
"io" "io"
"os/exec"
"time"
"git.sr.ht/~sircmpwn/aerc/commands"
"git.sr.ht/~sircmpwn/aerc/widgets" "git.sr.ht/~sircmpwn/aerc/widgets"
"github.com/gdamore/tcell"
) )
func init() { func init() {
@ -23,44 +20,13 @@ func Pipe(aerc *widgets.Aerc, args []string) error {
store := acct.Messages().Store() store := acct.Messages().Store()
msg := acct.Messages().Selected() msg := acct.Messages().Selected()
store.FetchFull([]uint32{msg.Uid}, func(reader io.Reader) { store.FetchFull([]uint32{msg.Uid}, func(reader io.Reader) {
cmd := exec.Command(args[1], args[2:]...) term, err := commands.QuickTerm(aerc, args[1:], reader)
pipe, err := cmd.StdinPipe()
if err != nil { if err != nil {
aerc.PushStatus(" "+err.Error(), 10*time.Second). aerc.PushError(" " + err.Error())
Color(tcell.ColorDefault, tcell.ColorRed)
return
}
term, err := widgets.NewTerminal(cmd)
if err != nil {
aerc.PushStatus(" "+err.Error(), 10*time.Second).
Color(tcell.ColorDefault, tcell.ColorRed)
return return
} }
name := args[1] + " <" + msg.Envelope.Subject name := args[1] + " <" + msg.Envelope.Subject
aerc.NewTab(term, name) aerc.NewTab(term, name)
term.OnClose = func(err error) {
if err != nil {
aerc.PushStatus(" "+err.Error(), 10*time.Second).
Color(tcell.ColorDefault, tcell.ColorRed)
} else {
aerc.PushStatus("Process complete, press any key to close.",
10*time.Second)
term.OnEvent = func(event tcell.Event) bool {
aerc.RemoveTab(term)
return true
}
}
}
term.OnStart = func() {
go func() {
_, err := io.Copy(pipe, reader)
if err != nil {
aerc.PushStatus(" "+err.Error(), 10*time.Second).
Color(tcell.ColorDefault, tcell.ColorRed)
}
pipe.Close()
}()
}
}) })
return nil return nil
} }

45
commands/msgview/pipe.go Normal file
View file

@ -0,0 +1,45 @@
package msgview
import (
"encoding/base64"
"errors"
"fmt"
"io"
"mime/quotedprintable"
"git.sr.ht/~sircmpwn/aerc/commands"
"git.sr.ht/~sircmpwn/aerc/widgets"
)
func init() {
register("pipe", Pipe)
}
func Pipe(aerc *widgets.Aerc, args []string) error {
if len(args) < 2 {
return errors.New("Usage: :pipe <cmd> [args...]")
}
mv := aerc.SelectedTab().(*widgets.MessageViewer)
p := mv.CurrentPart()
p.Store.FetchBodyPart(p.Msg.Uid, p.Index, func(reader io.Reader) {
// email parts are encoded as 7bit (plaintext), quoted-printable, or base64
switch p.Part.Encoding {
case "base64":
reader = base64.NewDecoder(base64.StdEncoding, reader)
case "quoted-printable":
reader = quotedprintable.NewReader(reader)
}
term, err := commands.QuickTerm(aerc, args[1:], reader)
if err != nil {
aerc.PushError(" " + err.Error())
return
}
name := fmt.Sprintf("%s <%s/[%d]", args[1], p.Msg.Envelope.Subject, p.Index)
aerc.NewTab(term, name)
})
return nil
}

59
commands/msgview/save.go Normal file
View file

@ -0,0 +1,59 @@
package msgview
import (
"encoding/base64"
"errors"
"io"
"mime/quotedprintable"
"os"
"time"
"git.sr.ht/~sircmpwn/aerc/widgets"
"github.com/mitchellh/go-homedir"
)
func init() {
register("save", Save)
}
func Save(aerc *widgets.Aerc, args []string) error {
if len(args) < 2 {
return errors.New("Usage: :save <path>")
}
mv := aerc.SelectedTab().(*widgets.MessageViewer)
p := mv.CurrentPart()
p.Store.FetchBodyPart(p.Msg.Uid, p.Index, func(reader io.Reader) {
// email parts are encoded as 7bit (plaintext), quoted-printable, or base64
switch p.Part.Encoding {
case "base64":
reader = base64.NewDecoder(base64.StdEncoding, reader)
case "quoted-printable":
reader = quotedprintable.NewReader(reader)
}
target, err := homedir.Expand(args[1])
if err != nil {
aerc.PushError(" " + err.Error())
return
}
f, err := os.Create(target)
if err != nil {
aerc.PushError(" " + err.Error())
return
}
defer f.Close()
_, err = io.Copy(f, reader)
if err != nil {
aerc.PushError(" " + err.Error())
return
}
aerc.PushStatus("Saved to "+target, 10*time.Second)
})
return nil
}

56
commands/util.go Normal file
View file

@ -0,0 +1,56 @@
package commands
import (
"io"
"os/exec"
"time"
"git.sr.ht/~sircmpwn/aerc/widgets"
"github.com/gdamore/tcell"
)
// QuickTerm is an ephemeral terminal for running a single command and quiting.
func QuickTerm(aerc *widgets.Aerc, args []string, stdin io.Reader) (*widgets.Terminal, error) {
cmd := exec.Command(args[0], args[1:]...)
pipe, err := cmd.StdinPipe()
if err != nil {
return nil, err
}
term, err := widgets.NewTerminal(cmd)
if err != nil {
return nil, err
}
term.OnClose = func(err error) {
if err != nil {
aerc.PushError(" " + err.Error())
// remove the tab on error, otherwise it gets stuck
aerc.RemoveTab(term)
} else {
aerc.PushStatus("Process complete, press any key to close.",
10*time.Second)
term.OnEvent = func(event tcell.Event) bool {
aerc.RemoveTab(term)
return true
}
}
}
term.OnStart = func() {
status := make(chan error, 1)
go func() {
_, err := io.Copy(pipe, stdin)
defer pipe.Close()
status <- err
}()
err := <-status
if err != nil {
aerc.PushError(" " + err.Error())
}
}
return term, nil
}

View file

@ -51,6 +51,7 @@ Rr = :reply -a<Enter>
Rq = :reply -aq<Enter> Rq = :reply -aq<Enter>
<C-k> = :prev-part<Enter> <C-k> = :prev-part<Enter>
<C-j> = :next-part<Enter> <C-j> = :next-part<Enter>
S = :save<space>
[compose] [compose]
# Keybindings used when the embedded terminal is not selected in the compose # Keybindings used when the embedded terminal is not selected in the compose

View file

@ -91,6 +91,13 @@ These commands work in any context.
## MESSAGE VIEW COMMANDS ## MESSAGE VIEW COMMANDS
*pipe* <cmd>
Downloads and pipes the current message part into the given shell command,
and opens a new terminal tab to show the result.
*save* <path>
Saves the current message part to the given path.
*close* *close*
Closes the message viewer. Closes the message viewer.

View file

@ -243,6 +243,10 @@ func (aerc *Aerc) PushStatus(text string, expiry time.Duration) *StatusMessage {
return aerc.statusline.Push(text, expiry) return aerc.statusline.Push(text, expiry)
} }
func (aerc *Aerc) PushError(text string) {
aerc.PushStatus(text, 10*time.Second).Color(tcell.ColorDefault, tcell.ColorRed)
}
func (aerc *Aerc) focus(item libui.Interactive) { func (aerc *Aerc) focus(item libui.Interactive) {
if aerc.focused == item { if aerc.focused == item {
return return

View file

@ -199,6 +199,18 @@ func (mv *MessageViewer) OnInvalidate(fn func(d ui.Drawable)) {
}) })
} }
func (mv *MessageViewer) CurrentPart() *PartInfo {
switcher := mv.switcher
part := switcher.parts[switcher.selected]
return &PartInfo{
Index: part.index,
Msg: part.msg,
Part: part.part,
Store: part.store,
}
}
func (mv *MessageViewer) PreviousPart() { func (mv *MessageViewer) PreviousPart() {
switcher := mv.switcher switcher := mv.switcher
for { for {
@ -291,6 +303,13 @@ type PartViewer struct {
term *Terminal term *Terminal
} }
type PartInfo struct {
Index []int
Msg *types.MessageInfo
Part *imap.BodyStructure
Store *lib.MessageStore
}
func NewPartViewer(conf *config.AercConfig, func NewPartViewer(conf *config.AercConfig,
store *lib.MessageStore, msg *types.MessageInfo, store *lib.MessageStore, msg *types.MessageInfo,
part *imap.BodyStructure, index []int) (*PartViewer, error) { part *imap.BodyStructure, index []int) (*PartViewer, error) {