2018-01-09 20:39:00 -05:00
|
|
|
package imap
|
|
|
|
|
|
|
|
import (
|
2018-01-31 21:54:52 -05:00
|
|
|
"crypto/tls"
|
|
|
|
"crypto/x509"
|
2018-01-14 11:30:11 +01:00
|
|
|
"fmt"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
2018-01-10 17:19:45 +01:00
|
|
|
|
2018-01-14 11:30:11 +01:00
|
|
|
"github.com/emersion/go-imap"
|
|
|
|
"github.com/emersion/go-imap-idle"
|
2018-01-31 21:18:21 -05:00
|
|
|
"github.com/emersion/go-imap/client"
|
|
|
|
|
|
|
|
"git.sr.ht/~sircmpwn/aerc2/worker/types"
|
2018-01-09 20:39:00 -05:00
|
|
|
)
|
|
|
|
|
2018-01-14 11:30:11 +01:00
|
|
|
var errUnsupported = fmt.Errorf("unsupported command")
|
|
|
|
|
|
|
|
type imapClient struct {
|
|
|
|
*client.Client
|
2019-05-13 20:16:55 -04:00
|
|
|
idle *idle.IdleClient
|
2018-01-14 11:30:11 +01:00
|
|
|
}
|
|
|
|
|
2018-01-09 20:39:00 -05:00
|
|
|
type IMAPWorker struct {
|
2018-01-14 11:30:11 +01:00
|
|
|
config struct {
|
|
|
|
scheme string
|
|
|
|
insecure bool
|
|
|
|
addr string
|
|
|
|
user *url.Userinfo
|
|
|
|
}
|
|
|
|
|
2019-03-10 23:45:00 -04:00
|
|
|
client *imapClient
|
2019-05-13 20:16:55 -04:00
|
|
|
idleStop chan struct{}
|
|
|
|
idleDone chan error
|
2019-03-10 23:45:00 -04:00
|
|
|
selected imap.MailboxStatus
|
|
|
|
updates chan client.Update
|
|
|
|
worker *types.Worker
|
2019-03-20 23:23:38 -04:00
|
|
|
// Map of sequence numbers to UIDs, index 0 is seq number 1
|
|
|
|
seqMap []uint32
|
2018-01-09 20:39:00 -05:00
|
|
|
}
|
|
|
|
|
2018-02-01 18:42:03 -05:00
|
|
|
func NewIMAPWorker(worker *types.Worker) *IMAPWorker {
|
2018-01-09 20:39:00 -05:00
|
|
|
return &IMAPWorker{
|
2019-05-13 20:16:55 -04:00
|
|
|
idleDone: make(chan error),
|
|
|
|
updates: make(chan client.Update, 50),
|
|
|
|
worker: worker,
|
2018-01-09 20:39:00 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-31 21:54:52 -05:00
|
|
|
func (w *IMAPWorker) verifyPeerCert(msg types.WorkerMessage) func(
|
|
|
|
rawCerts [][]byte, _ [][]*x509.Certificate) error {
|
|
|
|
|
|
|
|
return func(rawCerts [][]byte, _ [][]*x509.Certificate) error {
|
|
|
|
pool := x509.NewCertPool()
|
|
|
|
for _, rawCert := range rawCerts {
|
|
|
|
cert, err := x509.ParseCertificate(rawCert)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
pool.AddCert(cert)
|
|
|
|
}
|
|
|
|
|
2018-02-01 19:54:19 -05:00
|
|
|
request := &types.CertificateApprovalRequest{
|
2018-01-31 21:54:52 -05:00
|
|
|
Message: types.RespondTo(msg),
|
|
|
|
CertPool: pool,
|
|
|
|
}
|
2018-02-01 18:42:03 -05:00
|
|
|
w.worker.PostMessage(request, nil)
|
2018-01-31 21:54:52 -05:00
|
|
|
|
2018-02-01 18:42:03 -05:00
|
|
|
response := <-w.worker.Actions
|
2018-01-31 21:54:52 -05:00
|
|
|
if response.InResponseTo() != request {
|
2018-02-01 19:54:19 -05:00
|
|
|
return fmt.Errorf("Expected UI to respond to cert request")
|
2018-01-31 21:54:52 -05:00
|
|
|
}
|
2018-02-01 19:54:19 -05:00
|
|
|
if approval, ok := response.(*types.ApproveCertificate); !ok {
|
|
|
|
return fmt.Errorf("Expected UI to send certificate approval")
|
|
|
|
} else {
|
|
|
|
if approval.Approved {
|
|
|
|
return nil
|
|
|
|
} else {
|
|
|
|
return fmt.Errorf("UI rejected certificate")
|
|
|
|
}
|
2018-01-31 21:54:52 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-14 11:30:11 +01:00
|
|
|
func (w *IMAPWorker) handleMessage(msg types.WorkerMessage) error {
|
2019-05-13 20:16:55 -04:00
|
|
|
if w.idleStop != nil {
|
|
|
|
close(w.idleStop)
|
|
|
|
if err := <-w.idleDone; err != nil {
|
|
|
|
w.worker.PostMessage(&types.Error{Error: err}, nil)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-10 17:19:45 +01:00
|
|
|
switch msg := msg.(type) {
|
2018-02-01 19:54:19 -05:00
|
|
|
case *types.Unsupported:
|
2018-01-14 11:30:11 +01:00
|
|
|
// No-op
|
2018-02-01 19:54:19 -05:00
|
|
|
case *types.Configure:
|
2018-01-14 11:30:11 +01:00
|
|
|
u, err := url.Parse(msg.Config.Source)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2018-01-09 20:39:00 -05:00
|
|
|
}
|
2018-01-14 11:30:11 +01:00
|
|
|
|
|
|
|
w.config.scheme = u.Scheme
|
|
|
|
if strings.HasSuffix(w.config.scheme, "+insecure") {
|
|
|
|
w.config.scheme = strings.TrimSuffix(w.config.scheme, "+insecure")
|
|
|
|
w.config.insecure = true
|
|
|
|
}
|
|
|
|
|
|
|
|
w.config.addr = u.Host
|
|
|
|
if !strings.ContainsRune(w.config.addr, ':') {
|
|
|
|
w.config.addr += ":" + u.Scheme
|
2018-01-09 20:39:00 -05:00
|
|
|
}
|
2018-01-14 11:30:11 +01:00
|
|
|
|
|
|
|
w.config.scheme = u.Scheme
|
|
|
|
w.config.user = u.User
|
2018-02-01 19:54:19 -05:00
|
|
|
case *types.Connect:
|
2018-01-14 11:30:11 +01:00
|
|
|
var (
|
|
|
|
c *client.Client
|
|
|
|
err error
|
|
|
|
)
|
2018-01-31 21:54:52 -05:00
|
|
|
tlsConfig := &tls.Config{
|
|
|
|
InsecureSkipVerify: true,
|
2018-02-01 19:54:19 -05:00
|
|
|
VerifyPeerCertificate: w.verifyPeerCert(msg),
|
2018-01-31 21:54:52 -05:00
|
|
|
}
|
2018-01-14 11:30:11 +01:00
|
|
|
switch w.config.scheme {
|
|
|
|
case "imap":
|
|
|
|
c, err = client.Dial(w.config.addr)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !w.config.insecure {
|
2018-01-31 21:54:52 -05:00
|
|
|
if err := c.StartTLS(tlsConfig); err != nil {
|
2018-01-14 11:30:11 +01:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case "imaps":
|
2018-01-31 21:54:52 -05:00
|
|
|
c, err = client.DialTLS(w.config.addr, tlsConfig)
|
2018-01-14 11:30:11 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return fmt.Errorf("Unknown IMAP scheme %s", w.config.scheme)
|
|
|
|
}
|
|
|
|
|
|
|
|
if w.config.user != nil {
|
|
|
|
username := w.config.user.Username()
|
|
|
|
password, hasPassword := w.config.user.Password()
|
|
|
|
if !hasPassword {
|
|
|
|
// TODO: ask password
|
|
|
|
}
|
|
|
|
if err := c.Login(username, password); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := c.Select(imap.InboxName, false); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Updates = w.updates
|
|
|
|
w.client = &imapClient{c, idle.NewClient(c)}
|
2018-02-01 19:54:19 -05:00
|
|
|
w.worker.PostMessage(&types.Done{types.RespondTo(msg)}, nil)
|
|
|
|
case *types.ListDirectories:
|
2018-02-01 19:34:08 -05:00
|
|
|
w.handleListDirectories(msg)
|
2019-01-13 16:18:10 -05:00
|
|
|
case *types.OpenDirectory:
|
|
|
|
w.handleOpenDirectory(msg)
|
2019-03-10 23:45:00 -04:00
|
|
|
case *types.FetchDirectoryContents:
|
|
|
|
w.handleFetchDirectoryContents(msg)
|
2019-03-14 22:19:04 -04:00
|
|
|
case *types.FetchMessageHeaders:
|
|
|
|
w.handleFetchMessageHeaders(msg)
|
2019-03-31 12:14:37 -04:00
|
|
|
case *types.FetchMessageBodyPart:
|
|
|
|
w.handleFetchMessageBodyPart(msg)
|
2019-03-31 12:17:57 -04:00
|
|
|
case *types.FetchFullMessages:
|
|
|
|
w.handleFetchFullMessages(msg)
|
2019-03-20 23:23:38 -04:00
|
|
|
case *types.DeleteMessages:
|
|
|
|
w.handleDeleteMessages(msg)
|
2018-01-14 11:30:11 +01:00
|
|
|
default:
|
|
|
|
return errUnsupported
|
2018-01-09 20:39:00 -05:00
|
|
|
}
|
2019-05-13 20:16:55 -04:00
|
|
|
|
|
|
|
if w.idleStop != nil {
|
|
|
|
w.idleStop = make(chan struct{})
|
|
|
|
go func() {
|
|
|
|
w.idleDone <- w.client.idle.IdleWithFallback(w.idleStop, 0)
|
|
|
|
}()
|
|
|
|
}
|
2018-01-14 11:30:11 +01:00
|
|
|
return nil
|
2018-01-09 20:39:00 -05:00
|
|
|
}
|
|
|
|
|
2019-01-13 16:18:10 -05:00
|
|
|
func (w *IMAPWorker) handleImapUpdate(update client.Update) {
|
|
|
|
w.worker.Logger.Printf("(= %T", update)
|
|
|
|
switch update := update.(type) {
|
|
|
|
case *client.MailboxUpdate:
|
|
|
|
status := update.Mailbox
|
2019-03-10 23:45:00 -04:00
|
|
|
if w.selected.Name == status.Name {
|
|
|
|
w.selected = *status
|
|
|
|
}
|
2019-01-13 16:18:10 -05:00
|
|
|
w.worker.PostMessage(&types.DirectoryInfo{
|
|
|
|
Flags: status.Flags,
|
2019-01-13 19:37:06 -05:00
|
|
|
Name: status.Name,
|
|
|
|
ReadOnly: status.ReadOnly,
|
2019-01-13 16:18:10 -05:00
|
|
|
|
|
|
|
Exists: int(status.Messages),
|
|
|
|
Recent: int(status.Recent),
|
|
|
|
Unseen: int(status.Unseen),
|
|
|
|
}, nil)
|
2019-05-13 20:23:23 -04:00
|
|
|
case *client.ExpungeUpdate:
|
|
|
|
i := update.SeqNum - 1
|
|
|
|
uid := w.seqMap[i]
|
|
|
|
w.seqMap = append(w.seqMap[:i], w.seqMap[i+1:]...)
|
|
|
|
w.worker.PostMessage(&types.MessagesDeleted{
|
|
|
|
Uids: []uint32{uid},
|
|
|
|
}, nil)
|
2019-01-13 16:18:10 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-09 20:39:00 -05:00
|
|
|
func (w *IMAPWorker) Run() {
|
|
|
|
for {
|
|
|
|
select {
|
2018-02-01 18:42:03 -05:00
|
|
|
case msg := <-w.worker.Actions:
|
|
|
|
msg = w.worker.ProcessAction(msg)
|
2018-01-14 11:30:11 +01:00
|
|
|
if err := w.handleMessage(msg); err == errUnsupported {
|
2018-02-01 19:54:19 -05:00
|
|
|
w.worker.PostMessage(&types.Unsupported{
|
2018-01-14 11:30:11 +01:00
|
|
|
Message: types.RespondTo(msg),
|
2018-02-01 18:42:03 -05:00
|
|
|
}, nil)
|
2018-01-14 11:30:11 +01:00
|
|
|
} else if err != nil {
|
2018-02-01 19:54:19 -05:00
|
|
|
w.worker.PostMessage(&types.Error{
|
2018-01-14 11:30:11 +01:00
|
|
|
Message: types.RespondTo(msg),
|
|
|
|
Error: err,
|
2018-02-01 18:42:03 -05:00
|
|
|
}, nil)
|
2018-01-14 11:30:11 +01:00
|
|
|
}
|
|
|
|
case update := <-w.updates:
|
2019-01-13 16:18:10 -05:00
|
|
|
w.handleImapUpdate(update)
|
2018-01-09 20:39:00 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|