2019-06-02 07:15:04 +02:00
|
|
|
package msg
|
2019-05-16 18:15:34 +02:00
|
|
|
|
|
|
|
import (
|
2019-11-03 13:51:14 +01:00
|
|
|
"bytes"
|
2019-05-16 18:15:34 +02:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
2019-05-16 18:39:22 +02:00
|
|
|
"io"
|
2019-05-16 20:16:45 +02:00
|
|
|
gomail "net/mail"
|
2019-05-16 18:15:34 +02:00
|
|
|
"strings"
|
|
|
|
|
2019-05-18 21:34:16 +02:00
|
|
|
"git.sr.ht/~sircmpwn/getopt"
|
2019-05-16 20:09:57 +02:00
|
|
|
"github.com/emersion/go-message"
|
|
|
|
_ "github.com/emersion/go-message/charset"
|
|
|
|
"github.com/emersion/go-message/mail"
|
2019-05-16 18:15:34 +02:00
|
|
|
|
2019-07-08 04:43:58 +02:00
|
|
|
"git.sr.ht/~sircmpwn/aerc/models"
|
2019-05-18 02:57:10 +02:00
|
|
|
"git.sr.ht/~sircmpwn/aerc/widgets"
|
2019-05-16 18:15:34 +02:00
|
|
|
)
|
|
|
|
|
2019-06-27 19:33:11 +02:00
|
|
|
type reply struct{}
|
|
|
|
|
2019-05-16 18:15:34 +02:00
|
|
|
func init() {
|
2019-06-27 19:33:11 +02:00
|
|
|
register(reply{})
|
|
|
|
}
|
|
|
|
|
2019-09-03 21:34:03 +02:00
|
|
|
func (reply) Aliases() []string {
|
2019-08-18 11:33:13 +02:00
|
|
|
return []string{"reply"}
|
2019-06-27 19:33:11 +02:00
|
|
|
}
|
|
|
|
|
2019-09-03 21:34:03 +02:00
|
|
|
func (reply) Complete(aerc *widgets.Aerc, args []string) []string {
|
2019-06-27 19:33:11 +02:00
|
|
|
return nil
|
2019-05-16 18:15:34 +02:00
|
|
|
}
|
|
|
|
|
2019-09-03 21:34:03 +02:00
|
|
|
func (reply) Execute(aerc *widgets.Aerc, args []string) error {
|
2019-11-03 13:51:14 +01:00
|
|
|
opts, optind, err := getopt.Getopts(args, "aqT:")
|
2019-05-16 18:39:22 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-06-08 20:57:56 +02:00
|
|
|
if optind != len(args) {
|
2019-11-03 13:51:14 +01:00
|
|
|
return errors.New("Usage: reply [-aq -T <template>]")
|
2019-05-16 18:15:34 +02:00
|
|
|
}
|
2019-05-16 18:39:22 +02:00
|
|
|
var (
|
|
|
|
quote bool
|
|
|
|
replyAll bool
|
2019-11-03 13:51:14 +01:00
|
|
|
template string
|
2019-05-16 18:39:22 +02:00
|
|
|
)
|
|
|
|
for _, opt := range opts {
|
|
|
|
switch opt.Option {
|
|
|
|
case 'a':
|
|
|
|
replyAll = true
|
|
|
|
case 'q':
|
|
|
|
quote = true
|
2019-11-03 13:51:14 +01:00
|
|
|
case 'T':
|
|
|
|
template = opt.Value
|
2019-05-16 18:39:22 +02:00
|
|
|
}
|
|
|
|
}
|
2019-05-16 18:15:34 +02:00
|
|
|
|
2019-06-02 07:15:04 +02:00
|
|
|
widget := aerc.SelectedTab().(widgets.ProvidesMessage)
|
|
|
|
acct := widget.SelectedAccount()
|
2019-11-03 13:51:14 +01:00
|
|
|
|
2019-06-02 07:15:04 +02:00
|
|
|
if acct == nil {
|
|
|
|
return errors.New("No account selected")
|
|
|
|
}
|
2019-05-16 20:16:45 +02:00
|
|
|
conf := acct.AccountConfig()
|
|
|
|
us, _ := gomail.ParseAddress(conf.From)
|
2019-06-02 07:15:04 +02:00
|
|
|
store := widget.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 := widget.SelectedMessage()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-05-16 18:15:34 +02:00
|
|
|
acct.Logger().Println("Replying to email " + msg.Envelope.MessageId)
|
|
|
|
|
|
|
|
var (
|
|
|
|
to []string
|
|
|
|
cc []string
|
2019-07-08 04:43:58 +02:00
|
|
|
toList []*models.Address
|
2019-05-16 18:15:34 +02:00
|
|
|
)
|
2019-05-25 20:52:57 +02:00
|
|
|
if args[0] == "reply" {
|
|
|
|
if len(msg.Envelope.ReplyTo) != 0 {
|
|
|
|
toList = msg.Envelope.ReplyTo
|
2019-05-16 18:15:34 +02:00
|
|
|
} else {
|
2019-05-25 20:52:57 +02:00
|
|
|
toList = msg.Envelope.From
|
2019-05-16 18:15:34 +02:00
|
|
|
}
|
2019-05-25 20:52:57 +02:00
|
|
|
for _, addr := range toList {
|
2019-07-08 04:43:58 +02:00
|
|
|
if addr.Name != "" {
|
2019-05-25 20:52:57 +02:00
|
|
|
to = append(to, fmt.Sprintf("%s <%s@%s>",
|
2019-07-08 04:43:58 +02:00
|
|
|
addr.Name, addr.Mailbox, addr.Host))
|
2019-05-25 20:52:57 +02:00
|
|
|
} else {
|
2019-07-08 04:43:58 +02:00
|
|
|
to = append(to, fmt.Sprintf("<%s@%s>", addr.Mailbox, addr.Host))
|
2019-05-25 20:52:57 +02:00
|
|
|
}
|
2019-05-16 18:15:34 +02:00
|
|
|
}
|
2019-05-25 20:52:57 +02:00
|
|
|
if replyAll {
|
|
|
|
for _, addr := range msg.Envelope.Cc {
|
2019-07-08 04:43:58 +02:00
|
|
|
cc = append(cc, addr.Format())
|
2019-05-25 20:52:57 +02:00
|
|
|
}
|
|
|
|
for _, addr := range msg.Envelope.To {
|
2019-07-08 04:43:58 +02:00
|
|
|
address := fmt.Sprintf("%s@%s", addr.Mailbox, addr.Host)
|
2019-05-25 20:52:57 +02:00
|
|
|
if address == us.Address {
|
|
|
|
continue
|
|
|
|
}
|
2019-07-08 04:43:58 +02:00
|
|
|
to = append(to, addr.Format())
|
2019-05-16 20:16:45 +02:00
|
|
|
}
|
|
|
|
}
|
2019-05-16 18:15:34 +02:00
|
|
|
}
|
|
|
|
|
2019-05-16 18:42:46 +02:00
|
|
|
var subject string
|
2019-09-07 20:42:17 +02:00
|
|
|
if !strings.HasPrefix(strings.ToLower(msg.Envelope.Subject), "re: ") {
|
2019-08-18 11:33:13 +02:00
|
|
|
subject = "Re: " + msg.Envelope.Subject
|
2019-05-16 18:42:46 +02:00
|
|
|
} else {
|
2019-08-18 11:33:13 +02:00
|
|
|
subject = msg.Envelope.Subject
|
2019-05-16 18:42:46 +02:00
|
|
|
}
|
2019-05-16 18:15:34 +02:00
|
|
|
|
2019-07-23 01:29:07 +02:00
|
|
|
defaults := map[string]string{
|
|
|
|
"To": strings.Join(to, ", "),
|
|
|
|
"Cc": strings.Join(cc, ", "),
|
|
|
|
"Subject": subject,
|
|
|
|
"In-Reply-To": msg.Envelope.MessageId,
|
|
|
|
}
|
|
|
|
|
2019-11-03 13:51:14 +01:00
|
|
|
addTab := func() error {
|
|
|
|
if template != "" {
|
|
|
|
defaults["OriginalFrom"] = models.FormatAddresses(msg.Envelope.From)
|
|
|
|
defaults["OriginalDate"] = msg.Envelope.Date.Format("Mon Jan 2, 2006 at 3:04 PM")
|
|
|
|
}
|
2019-05-25 20:52:57 +02:00
|
|
|
|
2019-11-03 13:51:14 +01:00
|
|
|
composer, err := widgets.NewComposer(aerc, aerc.Config(),
|
|
|
|
acct.AccountConfig(), acct.Worker(), template, defaults)
|
|
|
|
if err != nil {
|
|
|
|
aerc.PushError("Error: " + err.Error())
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if args[0] == "reply" {
|
|
|
|
composer.FocusTerminal()
|
|
|
|
}
|
2019-05-16 18:15:34 +02:00
|
|
|
|
2019-05-16 18:39:22 +02:00
|
|
|
tab := aerc.NewTab(composer, subject)
|
2019-07-23 01:29:07 +02:00
|
|
|
composer.OnHeaderChange("Subject", func(subject string) {
|
2019-05-16 18:39:22 +02:00
|
|
|
if subject == "" {
|
|
|
|
tab.Name = "New email"
|
|
|
|
} else {
|
|
|
|
tab.Name = subject
|
|
|
|
}
|
|
|
|
tab.Content.Invalidate()
|
|
|
|
})
|
2019-11-03 13:51:14 +01:00
|
|
|
|
|
|
|
return nil
|
2019-05-16 18:39:22 +02:00
|
|
|
}
|
2019-05-16 18:15:34 +02:00
|
|
|
|
2019-08-18 11:33:13 +02:00
|
|
|
if quote {
|
2019-11-03 13:51:14 +01:00
|
|
|
if template == "" {
|
|
|
|
template = aerc.Config().Templates.QuotedReply
|
2019-08-18 11:33:13 +02:00
|
|
|
}
|
|
|
|
|
2019-11-03 13:51:14 +01:00
|
|
|
store.FetchBodyPart(msg.Uid, []int{1}, func(reader io.Reader) {
|
2019-05-16 20:09:57 +02:00
|
|
|
header := message.Header{}
|
2019-12-06 15:28:34 +01:00
|
|
|
if len(msg.BodyStructure.Parts) > 0 {
|
|
|
|
partID := 0 // TODO: will we always choose first msg part?
|
|
|
|
header.SetText(
|
|
|
|
"Content-Transfer-Encoding", msg.BodyStructure.Parts[partID].Encoding)
|
|
|
|
if msg.BodyStructure.Parts[partID].MIMESubType == "" {
|
|
|
|
header.SetContentType(
|
|
|
|
msg.BodyStructure.Parts[partID].MIMEType,
|
|
|
|
msg.BodyStructure.Parts[partID].Params)
|
|
|
|
} else {
|
|
|
|
// include SubType if defined (text/plain, text/html, ...)
|
|
|
|
header.SetContentType(
|
|
|
|
fmt.Sprintf("%s/%s", msg.BodyStructure.Parts[partID].MIMEType,
|
|
|
|
msg.BodyStructure.Parts[partID].MIMESubType),
|
|
|
|
msg.BodyStructure.Parts[partID].Params)
|
|
|
|
}
|
|
|
|
header.SetText("Content-Description", msg.BodyStructure.Parts[partID].Description)
|
|
|
|
} else { // Parts has no headers, so we use global headers info
|
|
|
|
header.SetText(
|
|
|
|
"Content-Transfer-Encoding", msg.BodyStructure.Encoding)
|
|
|
|
if msg.BodyStructure.MIMESubType == "" {
|
|
|
|
header.SetContentType(
|
|
|
|
msg.BodyStructure.MIMEType,
|
|
|
|
msg.BodyStructure.Params)
|
|
|
|
} else {
|
|
|
|
// include SubType if defined (text/plain, text/html, ...)
|
|
|
|
header.SetContentType(
|
|
|
|
fmt.Sprintf("%s/%s", msg.BodyStructure.MIMEType,
|
|
|
|
msg.BodyStructure.MIMESubType),
|
|
|
|
msg.BodyStructure.Params)
|
|
|
|
}
|
|
|
|
header.SetText("Content-Description", msg.BodyStructure.Description)
|
|
|
|
}
|
2019-05-16 20:09:57 +02:00
|
|
|
entity, err := message.New(header, reader)
|
|
|
|
if err != nil {
|
|
|
|
// TODO: Do something with the error
|
|
|
|
addTab()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
mreader := mail.NewReader(entity)
|
|
|
|
part, err := mreader.NextPart()
|
|
|
|
if err != nil {
|
|
|
|
// TODO: Do something with the error
|
|
|
|
addTab()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-11-03 13:51:14 +01:00
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
buf.ReadFrom(part.Body)
|
|
|
|
defaults["Original"] = buf.String()
|
2019-05-16 18:39:22 +02:00
|
|
|
addTab()
|
|
|
|
})
|
2019-11-03 13:51:14 +01:00
|
|
|
return nil
|
2019-05-16 18:39:22 +02:00
|
|
|
} else {
|
2019-11-03 13:51:14 +01:00
|
|
|
return addTab()
|
2019-05-16 18:39:22 +02:00
|
|
|
}
|
2019-05-16 18:15:34 +02:00
|
|
|
}
|
2019-06-02 01:47:09 +02:00
|
|
|
|
2019-07-08 04:43:58 +02:00
|
|
|
func findPlaintext(bs *models.BodyStructure,
|
|
|
|
path []int) (*models.BodyStructure, []int) {
|
2019-06-02 01:47:09 +02:00
|
|
|
|
|
|
|
for i, part := range bs.Parts {
|
|
|
|
cur := append(path, i+1)
|
2019-09-07 20:34:07 +02:00
|
|
|
if strings.ToLower(part.MIMEType) == "text" &&
|
|
|
|
strings.ToLower(part.MIMESubType) == "plain" {
|
2019-06-02 01:47:09 +02:00
|
|
|
return part, cur
|
|
|
|
}
|
2019-09-07 20:34:07 +02:00
|
|
|
if strings.ToLower(part.MIMEType) == "multipart" {
|
2019-06-02 15:36:21 +02:00
|
|
|
if part, path := findPlaintext(part, cur); path != nil {
|
2019-06-02 01:47:09 +02:00
|
|
|
return part, path
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil
|
|
|
|
}
|