2019-07-05 12:21:12 -04:00
|
|
|
package msg
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
2019-07-08 18:19:08 -04:00
|
|
|
"os/exec"
|
2020-05-28 10:32:32 -04:00
|
|
|
"time"
|
2019-07-05 12:21:12 -04:00
|
|
|
|
2021-11-05 10:19:46 +01:00
|
|
|
"git.sr.ht/~rjarry/aerc/commands"
|
|
|
|
"git.sr.ht/~rjarry/aerc/widgets"
|
|
|
|
"git.sr.ht/~rjarry/aerc/worker/types"
|
2019-07-08 18:19:08 -04:00
|
|
|
|
|
|
|
"git.sr.ht/~sircmpwn/getopt"
|
2019-07-05 12:21:12 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
type Pipe struct{}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
register(Pipe{})
|
|
|
|
}
|
|
|
|
|
2019-09-03 16:34:03 -03:00
|
|
|
func (Pipe) Aliases() []string {
|
2019-07-05 12:21:12 -04:00
|
|
|
return []string{"pipe"}
|
|
|
|
}
|
|
|
|
|
2019-09-03 16:34:03 -03:00
|
|
|
func (Pipe) Complete(aerc *widgets.Aerc, args []string) []string {
|
2019-07-05 12:21:12 -04:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-09-03 16:34:03 -03:00
|
|
|
func (Pipe) Execute(aerc *widgets.Aerc, args []string) error {
|
2019-07-05 12:21:12 -04:00
|
|
|
var (
|
2019-07-08 18:19:08 -04:00
|
|
|
background bool
|
|
|
|
pipeFull bool
|
|
|
|
pipePart bool
|
2019-07-05 12:21:12 -04:00
|
|
|
)
|
|
|
|
// TODO: let user specify part by index or preferred mimetype
|
2019-07-08 18:19:08 -04:00
|
|
|
opts, optind, err := getopt.Getopts(args, "bmp")
|
2019-07-05 12:21:12 -04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
for _, opt := range opts {
|
|
|
|
switch opt.Option {
|
2019-07-08 18:19:08 -04:00
|
|
|
case 'b':
|
|
|
|
background = true
|
2019-07-05 12:21:12 -04: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-08 18:19:08 -04:00
|
|
|
doTerm := func(reader io.Reader, name string) {
|
|
|
|
term, err := commands.QuickTerm(aerc, cmd, reader)
|
|
|
|
if err != nil {
|
2021-01-30 13:51:32 +01:00
|
|
|
aerc.PushError(err.Error())
|
2019-07-08 18:19:08 -04:00
|
|
|
return
|
|
|
|
}
|
|
|
|
aerc.NewTab(term, name)
|
|
|
|
}
|
|
|
|
|
|
|
|
doExec := func(reader io.Reader) {
|
|
|
|
ecmd := exec.Command(cmd[0], cmd[1:]...)
|
2019-07-08 18:50:40 -04:00
|
|
|
pipe, err := ecmd.StdinPipe()
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
defer pipe.Close()
|
|
|
|
io.Copy(pipe, reader)
|
|
|
|
}()
|
|
|
|
err = ecmd.Run()
|
2019-07-08 18:19:08 -04:00
|
|
|
if err != nil {
|
2021-01-30 13:51:32 +01:00
|
|
|
aerc.PushError(err.Error())
|
2019-07-08 18:19:08 -04:00
|
|
|
} else {
|
2019-07-08 18:32:31 -04:00
|
|
|
if ecmd.ProcessState.ExitCode() != 0 {
|
2020-07-27 01:03:55 -07:00
|
|
|
aerc.PushError(fmt.Sprintf(
|
|
|
|
"%s: completed with status %d", cmd[0],
|
|
|
|
ecmd.ProcessState.ExitCode()))
|
|
|
|
} else {
|
|
|
|
aerc.PushStatus(fmt.Sprintf(
|
|
|
|
"%s: completed with status %d", cmd[0],
|
|
|
|
ecmd.ProcessState.ExitCode()), 10*time.Second)
|
2019-07-08 18:32:31 -04:00
|
|
|
}
|
2019-07-08 18:19:08 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-05 12:21:12 -04:00
|
|
|
if pipeFull {
|
|
|
|
store := provider.Store()
|
2019-07-14 00:42:24 -07:00
|
|
|
if store == nil {
|
|
|
|
return errors.New("Cannot perform action. Messages still loading")
|
|
|
|
}
|
2019-07-09 17:04:21 -07: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-08 18:19:08 -04:00
|
|
|
if background {
|
2020-04-24 22:31:39 +02:00
|
|
|
doExec(fm.Content.Reader)
|
2019-07-08 18:19:08 -04:00
|
|
|
} else {
|
2020-04-24 22:31:39 +02:00
|
|
|
doTerm(fm.Content.Reader, fmt.Sprintf(
|
2019-07-08 18:19:08 -04:00
|
|
|
"%s <%s", cmd[0], msg.Envelope.Subject))
|
2019-07-05 12:21:12 -04: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 10:33:44 -05: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-08 18:19:08 -04: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 12:21:12 -04:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|