2019-06-02 05:15:04 +00:00
|
|
|
package msg
|
2019-05-16 16:15:34 +00:00
|
|
|
|
|
|
|
import (
|
2019-11-03 12:51:14 +00:00
|
|
|
"bytes"
|
2019-05-16 16:15:34 +00:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
2019-05-16 16:39:22 +00:00
|
|
|
"io"
|
2022-02-28 17:02:28 +00:00
|
|
|
"regexp"
|
2019-05-16 16:15:34 +00:00
|
|
|
"strings"
|
|
|
|
|
2019-05-18 19:34:16 +00:00
|
|
|
"git.sr.ht/~sircmpwn/getopt"
|
2019-05-16 16:15:34 +00:00
|
|
|
|
2021-11-05 09:19:46 +00:00
|
|
|
"git.sr.ht/~rjarry/aerc/lib"
|
|
|
|
"git.sr.ht/~rjarry/aerc/lib/format"
|
|
|
|
"git.sr.ht/~rjarry/aerc/models"
|
|
|
|
"git.sr.ht/~rjarry/aerc/widgets"
|
2020-11-10 18:57:09 +00:00
|
|
|
"github.com/emersion/go-message/mail"
|
2019-05-16 16:15:34 +00:00
|
|
|
)
|
|
|
|
|
2019-06-27 17:33:11 +00:00
|
|
|
type reply struct{}
|
|
|
|
|
2019-05-16 16:15:34 +00:00
|
|
|
func init() {
|
2019-06-27 17:33:11 +00:00
|
|
|
register(reply{})
|
|
|
|
}
|
|
|
|
|
2019-09-03 19:34:03 +00:00
|
|
|
func (reply) Aliases() []string {
|
2019-08-18 09:33:13 +00:00
|
|
|
return []string{"reply"}
|
2019-06-27 17:33:11 +00:00
|
|
|
}
|
|
|
|
|
2019-09-03 19:34:03 +00:00
|
|
|
func (reply) Complete(aerc *widgets.Aerc, args []string) []string {
|
2019-06-27 17:33:11 +00:00
|
|
|
return nil
|
2019-05-16 16:15:34 +00:00
|
|
|
}
|
|
|
|
|
2019-09-03 19:34:03 +00:00
|
|
|
func (reply) Execute(aerc *widgets.Aerc, args []string) error {
|
2019-11-03 12:51:14 +00:00
|
|
|
opts, optind, err := getopt.Getopts(args, "aqT:")
|
2019-05-16 16:39:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-06-08 18:57:56 +00:00
|
|
|
if optind != len(args) {
|
2019-11-03 12:51:14 +00:00
|
|
|
return errors.New("Usage: reply [-aq -T <template>]")
|
2019-05-16 16:15:34 +00:00
|
|
|
}
|
2019-05-16 16:39:22 +00:00
|
|
|
var (
|
|
|
|
quote bool
|
|
|
|
replyAll bool
|
2019-11-03 12:51:14 +00:00
|
|
|
template string
|
2019-05-16 16:39:22 +00:00
|
|
|
)
|
|
|
|
for _, opt := range opts {
|
|
|
|
switch opt.Option {
|
|
|
|
case 'a':
|
|
|
|
replyAll = true
|
|
|
|
case 'q':
|
|
|
|
quote = true
|
2019-11-03 12:51:14 +00:00
|
|
|
case 'T':
|
|
|
|
template = opt.Value
|
2019-05-16 16:39:22 +00:00
|
|
|
}
|
|
|
|
}
|
2019-05-16 16:15:34 +00:00
|
|
|
|
2022-07-18 10:54:55 +00:00
|
|
|
widget := aerc.SelectedTabContent().(widgets.ProvidesMessage)
|
2019-06-02 05:15:04 +00:00
|
|
|
acct := widget.SelectedAccount()
|
2019-11-03 12:51:14 +00:00
|
|
|
|
2019-06-02 05:15:04 +00:00
|
|
|
if acct == nil {
|
|
|
|
return errors.New("No account selected")
|
|
|
|
}
|
2019-05-16 18:16:45 +00:00
|
|
|
conf := acct.AccountConfig()
|
2020-11-10 19:35:47 +00:00
|
|
|
from, err := mail.ParseAddress(conf.From)
|
2020-08-19 10:06:02 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-11-10 19:35:47 +00:00
|
|
|
var aliases []*mail.Address
|
|
|
|
if conf.Aliases != "" {
|
|
|
|
aliases, err = mail.ParseAddressList(conf.Aliases)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-08-20 17:22:50 +00:00
|
|
|
}
|
2020-08-26 06:13:38 +00:00
|
|
|
|
2019-06-02 05:15:04 +00:00
|
|
|
store := widget.Store()
|
2019-07-14 07:42:24 +00:00
|
|
|
if store == nil {
|
|
|
|
return errors.New("Cannot perform action. Messages still loading")
|
|
|
|
}
|
2019-07-10 00:04:21 +00:00
|
|
|
msg, err := widget.SelectedMessage()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-05-16 16:15:34 +00:00
|
|
|
|
2020-08-26 06:13:38 +00:00
|
|
|
// figure out the sending from address if we have aliases
|
|
|
|
if len(aliases) != 0 {
|
|
|
|
rec := newAddrSet()
|
|
|
|
rec.AddList(msg.Envelope.To)
|
|
|
|
rec.AddList(msg.Envelope.Cc)
|
|
|
|
// test the from first, it has priority over any present alias
|
|
|
|
if rec.Contains(from) {
|
|
|
|
// do nothing
|
|
|
|
} else {
|
|
|
|
for _, a := range aliases {
|
|
|
|
if rec.Contains(a) {
|
|
|
|
from = a
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-16 16:15:34 +00:00
|
|
|
var (
|
2020-11-10 18:57:09 +00:00
|
|
|
to []*mail.Address
|
|
|
|
cc []*mail.Address
|
2019-05-16 16:15:34 +00:00
|
|
|
)
|
2020-08-20 17:22:50 +00:00
|
|
|
|
2020-08-26 06:13:38 +00:00
|
|
|
recSet := newAddrSet() // used for de-duping
|
2020-08-20 17:22:50 +00:00
|
|
|
|
2020-08-26 06:13:38 +00:00
|
|
|
if len(msg.Envelope.ReplyTo) != 0 {
|
|
|
|
to = msg.Envelope.ReplyTo
|
|
|
|
} else {
|
|
|
|
to = msg.Envelope.From
|
|
|
|
}
|
2022-01-31 14:28:58 +00:00
|
|
|
|
|
|
|
if !aerc.Config().Compose.ReplyToSelf {
|
|
|
|
for i, v := range to {
|
|
|
|
if v.Address == from.Address {
|
|
|
|
to = append(to[:i], to[i+1:]...)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(to) == 0 {
|
2022-03-09 21:48:00 +00:00
|
|
|
to = msg.Envelope.To
|
2022-01-31 14:28:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-26 06:13:38 +00:00
|
|
|
recSet.AddList(to)
|
2020-08-20 17:22:50 +00:00
|
|
|
|
2020-08-26 06:13:38 +00:00
|
|
|
if replyAll {
|
|
|
|
// order matters, due to the deduping
|
|
|
|
// in order of importance, first parse the To, then the Cc header
|
|
|
|
|
|
|
|
// we add our from address, so that we don't self address ourselves
|
|
|
|
recSet.Add(from)
|
|
|
|
|
2020-11-10 18:57:09 +00:00
|
|
|
envTos := make([]*mail.Address, 0, len(msg.Envelope.To))
|
2020-08-26 06:13:38 +00:00
|
|
|
for _, addr := range msg.Envelope.To {
|
|
|
|
if recSet.Contains(addr) {
|
|
|
|
continue
|
2020-05-04 17:17:23 +00:00
|
|
|
}
|
2020-08-26 06:13:38 +00:00
|
|
|
envTos = append(envTos, addr)
|
2020-05-04 17:17:23 +00:00
|
|
|
}
|
2020-08-26 06:13:38 +00:00
|
|
|
recSet.AddList(envTos)
|
|
|
|
to = append(to, envTos...)
|
|
|
|
|
|
|
|
for _, addr := range msg.Envelope.Cc {
|
|
|
|
//dedupe stuff from the to/from headers
|
|
|
|
if recSet.Contains(addr) {
|
|
|
|
continue
|
2019-05-16 18:16:45 +00:00
|
|
|
}
|
2020-08-26 06:13:38 +00:00
|
|
|
cc = append(cc, addr)
|
2019-05-16 18:16:45 +00:00
|
|
|
}
|
2020-08-26 06:13:38 +00:00
|
|
|
recSet.AddList(cc)
|
2019-05-16 16:15:34 +00:00
|
|
|
}
|
|
|
|
|
2022-02-28 17:02:28 +00:00
|
|
|
subject := "Re: " + trimLocalizedRe(msg.Envelope.Subject)
|
2019-05-16 16:15:34 +00:00
|
|
|
|
2020-11-10 19:27:30 +00:00
|
|
|
h := &mail.Header{}
|
|
|
|
h.SetAddressList("to", to)
|
|
|
|
h.SetAddressList("cc", cc)
|
|
|
|
h.SetAddressList("from", []*mail.Address{from})
|
|
|
|
h.SetSubject(subject)
|
|
|
|
h.SetMsgIDList("in-reply-to", []string{msg.Envelope.MessageId})
|
2020-11-08 15:15:26 +00:00
|
|
|
err = setReferencesHeader(h, msg.RFC822Headers)
|
|
|
|
if err != nil {
|
|
|
|
aerc.PushError(fmt.Sprintf("could not set references: %v", err))
|
|
|
|
}
|
2020-11-03 06:39:36 +00:00
|
|
|
original := models.OriginalMail{
|
2020-11-10 19:27:30 +00:00
|
|
|
From: format.FormatAddresses(msg.Envelope.From),
|
|
|
|
Date: msg.Envelope.Date,
|
2020-11-03 06:39:36 +00:00
|
|
|
RFC822Headers: msg.RFC822Headers,
|
|
|
|
}
|
2019-07-22 23:29:07 +00:00
|
|
|
|
2019-11-03 12:51:14 +00:00
|
|
|
addTab := func() error {
|
2020-04-24 09:42:21 +00:00
|
|
|
composer, err := widgets.NewComposer(aerc, acct, aerc.Config(),
|
2020-11-10 19:27:30 +00:00
|
|
|
acct.AccountConfig(), acct.Worker(), template, h, original)
|
2019-11-03 12:51:14 +00:00
|
|
|
if err != nil {
|
2020-05-28 14:32:42 +00:00
|
|
|
aerc.PushError("Error: " + err.Error())
|
2019-11-03 12:51:14 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if args[0] == "reply" {
|
|
|
|
composer.FocusTerminal()
|
|
|
|
}
|
2019-05-16 16:15:34 +00:00
|
|
|
|
2019-05-16 16:39:22 +00:00
|
|
|
tab := aerc.NewTab(composer, subject)
|
2019-07-22 23:29:07 +00:00
|
|
|
composer.OnHeaderChange("Subject", func(subject string) {
|
2019-05-16 16:39:22 +00:00
|
|
|
if subject == "" {
|
|
|
|
tab.Name = "New email"
|
|
|
|
} else {
|
|
|
|
tab.Name = subject
|
|
|
|
}
|
|
|
|
tab.Content.Invalidate()
|
|
|
|
})
|
2019-11-03 12:51:14 +00:00
|
|
|
|
2020-05-25 14:59:48 +00:00
|
|
|
composer.OnClose(func(c *widgets.Composer) {
|
2020-09-21 18:43:22 +00:00
|
|
|
if c.Sent() {
|
|
|
|
store.Answered([]uint32{msg.Uid}, true, nil)
|
|
|
|
}
|
2020-05-25 14:59:48 +00:00
|
|
|
})
|
|
|
|
|
2019-11-03 12:51:14 +00:00
|
|
|
return nil
|
2019-05-16 16:39:22 +00:00
|
|
|
}
|
2019-05-16 16:15:34 +00:00
|
|
|
|
2019-08-18 09:33:13 +00:00
|
|
|
if quote {
|
2019-11-03 12:51:14 +00:00
|
|
|
if template == "" {
|
|
|
|
template = aerc.Config().Templates.QuotedReply
|
2019-08-18 09:33:13 +00:00
|
|
|
}
|
|
|
|
|
2020-06-19 15:58:08 +00:00
|
|
|
part := lib.FindPlaintext(msg.BodyStructure, nil)
|
2020-05-17 10:08:17 +00:00
|
|
|
if part == nil {
|
2020-06-19 15:58:08 +00:00
|
|
|
// mkey... let's get the first thing that isn't a container
|
|
|
|
// if that's still nil it's either not a multipart msg (ok) or
|
|
|
|
// broken (containers only)
|
|
|
|
part = lib.FindFirstNonMultipart(msg.BodyStructure, nil)
|
2020-05-17 10:08:17 +00:00
|
|
|
}
|
2021-02-26 21:08:49 +00:00
|
|
|
|
|
|
|
err = addMimeType(msg, part, &original)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-05-17 10:08:17 +00:00
|
|
|
store.FetchBodyPart(msg.Uid, part, func(reader io.Reader) {
|
2019-11-03 12:51:14 +00:00
|
|
|
buf := new(bytes.Buffer)
|
2020-01-04 20:13:51 +00:00
|
|
|
buf.ReadFrom(reader)
|
2020-01-08 20:44:14 +00:00
|
|
|
original.Text = buf.String()
|
2019-05-16 16:39:22 +00:00
|
|
|
addTab()
|
|
|
|
})
|
2019-11-03 12:51:14 +00:00
|
|
|
return nil
|
2019-05-16 16:39:22 +00:00
|
|
|
} else {
|
2022-01-27 08:16:54 +00:00
|
|
|
if template == "" {
|
|
|
|
template = aerc.Config().Templates.NewMessage
|
|
|
|
}
|
2019-11-03 12:51:14 +00:00
|
|
|
return addTab()
|
2019-05-16 16:39:22 +00:00
|
|
|
}
|
2019-05-16 16:15:34 +00:00
|
|
|
}
|
2020-08-26 06:13:38 +00:00
|
|
|
|
|
|
|
type addrSet map[string]struct{}
|
|
|
|
|
|
|
|
func newAddrSet() addrSet {
|
|
|
|
s := make(map[string]struct{})
|
|
|
|
return addrSet(s)
|
|
|
|
}
|
|
|
|
|
2020-11-10 18:57:09 +00:00
|
|
|
func (s addrSet) Add(a *mail.Address) {
|
2020-08-26 06:13:38 +00:00
|
|
|
s[a.Address] = struct{}{}
|
|
|
|
}
|
|
|
|
|
2020-11-10 18:57:09 +00:00
|
|
|
func (s addrSet) AddList(al []*mail.Address) {
|
2020-08-26 06:13:38 +00:00
|
|
|
for _, a := range al {
|
|
|
|
s[a.Address] = struct{}{}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-10 18:57:09 +00:00
|
|
|
func (s addrSet) Contains(a *mail.Address) bool {
|
2020-08-26 06:13:38 +00:00
|
|
|
_, ok := s[a.Address]
|
|
|
|
return ok
|
|
|
|
}
|
2020-11-08 15:15:26 +00:00
|
|
|
|
|
|
|
//setReferencesHeader adds the references header to target based on parent
|
|
|
|
//according to RFC2822
|
|
|
|
func setReferencesHeader(target, parent *mail.Header) error {
|
|
|
|
refs, err := parent.MsgIDList("references")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(refs) == 0 {
|
|
|
|
// according to the RFC we need to fall back to in-reply-to only if
|
|
|
|
// References is not set
|
|
|
|
refs, err = parent.MsgIDList("in-reply-to")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
msgID, err := parent.MessageID()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
refs = append(refs, msgID)
|
|
|
|
target.SetMsgIDList("references", refs)
|
|
|
|
return nil
|
|
|
|
}
|
2021-02-26 21:08:49 +00:00
|
|
|
|
|
|
|
// addMimeType adds the proper mime type of the part to the originalMail struct
|
|
|
|
func addMimeType(msg *models.MessageInfo, part []int,
|
|
|
|
orig *models.OriginalMail) error {
|
|
|
|
// caution, :forward uses the code as well, keep that in mind when modifying
|
|
|
|
bs, err := msg.BodyStructure.PartAtIndex(part)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
orig.MIMEType = fmt.Sprintf("%s/%s", bs.MIMEType, bs.MIMESubType)
|
|
|
|
return nil
|
|
|
|
}
|
2022-02-28 17:02:28 +00:00
|
|
|
|
|
|
|
// trimLocalizedRe removes known localizations of Re: commonly used by Outlook.
|
|
|
|
func trimLocalizedRe(subject string) string {
|
|
|
|
return strings.TrimPrefix(subject, localizedRe.FindString(subject))
|
|
|
|
}
|
|
|
|
|
|
|
|
// localizedRe contains a list of known translations for the common Re:
|
|
|
|
var localizedRe = regexp.MustCompile(`(?i)^((AW|RE|SV|VS|ODP|R): ?)+`)
|