2019-07-05 18:21:12 +02:00
|
|
|
package msg
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
2019-07-09 00:19:08 +02:00
|
|
|
"os/exec"
|
2020-05-28 16:32:32 +02:00
|
|
|
"time"
|
2019-07-05 18:21:12 +02:00
|
|
|
|
|
|
|
"git.sr.ht/~sircmpwn/aerc/commands"
|
|
|
|
"git.sr.ht/~sircmpwn/aerc/widgets"
|
2020-04-24 22:31:39 +02:00
|
|
|
"git.sr.ht/~sircmpwn/aerc/worker/types"
|
2019-07-09 00:19:08 +02:00
|
|
|
|
|
|
|
"git.sr.ht/~sircmpwn/getopt"
|
2020-07-30 23:22:32 +02:00
|
|
|
"github.com/gdamore/tcell"
|
2019-07-05 18:21:12 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type Pipe struct{}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
register(Pipe{})
|
|
|
|
}
|
|
|
|
|
2019-09-03 21:34:03 +02:00
|
|
|
func (Pipe) Aliases() []string {
|
2019-07-05 18:21:12 +02:00
|
|
|
return []string{"pipe"}
|
|
|
|
}
|
|
|
|
|
2019-09-03 21:34:03 +02:00
|
|
|
func (Pipe) Complete(aerc *widgets.Aerc, args []string) []string {
|
2019-07-05 18:21:12 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-09-03 21:34:03 +02:00
|
|
|
func (Pipe) Execute(aerc *widgets.Aerc, args []string) error {
|
2019-07-05 18:21:12 +02:00
|
|
|
var (
|
2019-07-09 00:19:08 +02:00
|
|
|
background bool
|
|
|
|
pipeFull bool
|
|
|
|
pipePart bool
|
2019-07-05 18:21:12 +02:00
|
|
|
)
|
|
|
|
// TODO: let user specify part by index or preferred mimetype
|
2019-07-09 00:19:08 +02:00
|
|
|
opts, optind, err := getopt.Getopts(args, "bmp")
|
2019-07-05 18:21:12 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
for _, opt := range opts {
|
|
|
|
switch opt.Option {
|
2019-07-09 00:19:08 +02:00
|
|
|
case 'b':
|
|
|
|
background = true
|
2019-07-05 18:21:12 +02:00
|
|
|
case 'm':
|
|
|
|
if pipePart {
|
|
|
|
return errors.New("-m and -p are mutually exclusive")
|
|
|
|
}
|
|
|
|
pipeFull = true
|
|
|
|
case 'p':
|
|
|
|
if pipeFull {
|
|
|
|
return errors.New("-m and -p are mutually exclusive")
|
|
|
|
}
|
|
|
|
pipePart = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
cmd := args[optind:]
|
|
|
|
if len(cmd) == 0 {
|
|
|
|
return errors.New("Usage: pipe [-mp] <cmd> [args...]")
|
|
|
|
}
|
|
|
|
|
|
|
|
provider := aerc.SelectedTab().(widgets.ProvidesMessage)
|
|
|
|
if !pipeFull && !pipePart {
|
|
|
|
if _, ok := provider.(*widgets.MessageViewer); ok {
|
|
|
|
pipePart = true
|
|
|
|
} else if _, ok := provider.(*widgets.AccountView); ok {
|
|
|
|
pipeFull = true
|
|
|
|
} else {
|
|
|
|
return errors.New(
|
|
|
|
"Neither -m nor -p specified and cannot infer default")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-09 00:19:08 +02:00
|
|
|
doTerm := func(reader io.Reader, name string) {
|
|
|
|
term, err := commands.QuickTerm(aerc, cmd, reader)
|
|
|
|
if err != nil {
|
2020-05-28 16:32:42 +02:00
|
|
|
aerc.PushError(" " + err.Error())
|
2019-07-09 00:19:08 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
aerc.NewTab(term, name)
|
|
|
|
}
|
|
|
|
|
|
|
|
doExec := func(reader io.Reader) {
|
|
|
|
ecmd := exec.Command(cmd[0], cmd[1:]...)
|
2019-07-09 00:50:40 +02:00
|
|
|
pipe, err := ecmd.StdinPipe()
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
defer pipe.Close()
|
|
|
|
io.Copy(pipe, reader)
|
|
|
|
}()
|
|
|
|
err = ecmd.Run()
|
2019-07-09 00:19:08 +02:00
|
|
|
if err != nil {
|
2020-05-28 16:32:42 +02:00
|
|
|
aerc.PushError(" " + err.Error())
|
2019-07-09 00:19:08 +02:00
|
|
|
} else {
|
2020-07-30 23:22:32 +02:00
|
|
|
color := tcell.ColorDefault
|
2019-07-09 00:32:31 +02:00
|
|
|
if ecmd.ProcessState.ExitCode() != 0 {
|
2020-07-30 23:22:32 +02:00
|
|
|
color = tcell.ColorRed
|
2019-07-09 00:32:31 +02:00
|
|
|
}
|
2020-07-30 23:22:32 +02:00
|
|
|
aerc.PushStatus(fmt.Sprintf(
|
|
|
|
"%s: completed with status %d", cmd[0],
|
|
|
|
ecmd.ProcessState.ExitCode()), 10*time.Second).
|
|
|
|
Color(tcell.ColorDefault, color)
|
2019-07-09 00:19:08 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-05 18:21:12 +02:00
|
|
|
if pipeFull {
|
|
|
|
store := provider.Store()
|
2019-07-14 09:42:24 +02:00
|
|
|
if store == nil {
|
|
|
|
return errors.New("Cannot perform action. Messages still loading")
|
|
|
|
}
|
2019-07-10 02:04:21 +02:00
|
|
|
msg, err := provider.SelectedMessage()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-04-24 22:31:39 +02:00
|
|
|
store.FetchFull([]uint32{msg.Uid}, func(fm *types.FullMessage) {
|
2019-07-09 00:19:08 +02:00
|
|
|
if background {
|
2020-04-24 22:31:39 +02:00
|
|
|
doExec(fm.Content.Reader)
|
2019-07-09 00:19:08 +02:00
|
|
|
} else {
|
2020-04-24 22:31:39 +02:00
|
|
|
doTerm(fm.Content.Reader, fmt.Sprintf(
|
2019-07-09 00:19:08 +02:00
|
|
|
"%s <%s", cmd[0], msg.Envelope.Subject))
|
2019-07-05 18:21:12 +02:00
|
|
|
}
|
|
|
|
})
|
|
|
|
} else if pipePart {
|
|
|
|
p := provider.SelectedMessagePart()
|
2020-07-05 14:27:21 +02:00
|
|
|
if p == nil {
|
|
|
|
return fmt.Errorf("could not fetch message part")
|
|
|
|
}
|
2020-03-06 16:33:44 +01:00
|
|
|
store := provider.Store()
|
2020-05-17 11:44:38 +02:00
|
|
|
store.FetchBodyPart(p.Msg.Uid, p.Index, func(reader io.Reader) {
|
2019-07-09 00:19:08 +02:00
|
|
|
if background {
|
|
|
|
doExec(reader)
|
|
|
|
} else {
|
|
|
|
name := fmt.Sprintf("%s <%s/[%d]",
|
|
|
|
cmd[0], p.Msg.Envelope.Subject, p.Index)
|
|
|
|
doTerm(reader, name)
|
2019-07-05 18:21:12 +02:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|