gte/pkg/mstore/imap.go

263 lines
5.4 KiB
Go
Raw Normal View History

2021-01-26 07:16:53 +01:00
package mstore
import (
2021-05-13 08:15:14 +02:00
"errors"
2021-01-27 10:07:31 +01:00
"fmt"
2021-01-29 12:29:23 +01:00
"io"
2021-01-29 17:22:07 +01:00
"io/ioutil"
2021-01-29 12:29:23 +01:00
"strings"
2021-01-27 15:01:42 +01:00
"time"
2021-01-27 10:07:31 +01:00
"github.com/emersion/go-imap"
2021-01-26 07:16:53 +01:00
"github.com/emersion/go-imap/client"
2021-01-29 17:22:07 +01:00
"github.com/emersion/go-message/mail"
2021-01-26 07:16:53 +01:00
)
2021-05-13 08:15:14 +02:00
var (
ErrIMAPInvalidConfig = errors.New("invalid imap configuration")
ErrIMAPConnFailure = errors.New("could not connect with imap")
ErrIMAPNotConnected = errors.New("unable to perform, not connected to imap")
ErrIMAPServerProblem = errors.New("imap server was unable to perform operation")
)
type IMAPBody struct {
2021-01-29 12:29:23 +01:00
reader io.Reader
length int
}
2021-05-13 08:15:14 +02:00
func NewIMAPBody(msg string) *IMAPBody {
return &IMAPBody{
2021-01-29 12:29:23 +01:00
reader: strings.NewReader(msg),
length: len([]byte(msg)),
}
}
2021-05-13 08:15:14 +02:00
func (b *IMAPBody) Read(p []byte) (int, error) {
2021-01-29 12:29:23 +01:00
return b.reader.Read(p)
}
2021-05-13 08:15:14 +02:00
func (b *IMAPBody) Len() int {
2021-01-29 12:29:23 +01:00
return b.length
}
2021-05-13 08:15:14 +02:00
type IMAPConfig struct {
IMAPURL string
IMAPUsername string
IMAPPassword string
IMAPFolderPrefix string
2021-01-26 07:16:53 +01:00
}
2021-05-13 08:15:14 +02:00
func (esc *IMAPConfig) Valid() bool {
if esc.IMAPURL == "" {
2021-01-27 10:07:31 +01:00
return false
}
2021-05-13 08:15:14 +02:00
if esc.IMAPUsername == "" || esc.IMAPPassword == "" {
2021-01-27 10:07:31 +01:00
return false
}
return true
}
2021-05-13 08:15:14 +02:00
type IMAP struct {
config *IMAPConfig
connected bool
client *client.Client
2021-01-28 09:40:08 +01:00
mboxStatus *imap.MailboxStatus
2021-01-27 12:58:43 +01:00
}
2021-05-13 08:15:14 +02:00
func NewIMAP(config *IMAPConfig) *IMAP {
return &IMAP{
config: config,
}
}
func (im *IMAP) Connect() error {
if !im.config.Valid() {
return ErrIMAPInvalidConfig
}
if im.connected {
return nil
}
cl, err := client.DialTLS(im.config.IMAPURL, nil)
2021-01-27 10:07:31 +01:00
if err != nil {
2021-05-13 08:15:14 +02:00
return fmt.Errorf("%w: %v", ErrIMAPConnFailure, err)
2021-01-27 10:07:31 +01:00
}
2021-05-13 08:15:14 +02:00
if err := cl.Login(im.config.IMAPUsername, im.config.IMAPPassword); err != nil {
return fmt.Errorf("%w: %v", ErrIMAPConnFailure, err)
2021-01-27 10:07:31 +01:00
}
2021-05-13 08:15:14 +02:00
im.client = cl
im.connected = true
return nil
2021-01-27 10:07:31 +01:00
}
2021-05-13 08:15:14 +02:00
func (im *IMAP) Close() {
im.client.Logout()
im.connected = false
2021-01-27 10:07:31 +01:00
}
2021-05-13 08:15:14 +02:00
func (im *IMAP) selectFolder(folder string) error {
if !im.connected {
return ErrIMAPNotConnected
}
folder = fmt.Sprintf("%s%s", im.config.IMAPFolderPrefix, folder)
2021-05-13 08:15:14 +02:00
status, err := im.client.Select(folder, false)
2021-01-27 12:58:43 +01:00
if err != nil {
2021-05-13 08:15:14 +02:00
return fmt.Errorf("%w, %v", ErrIMAPServerProblem, err)
2021-01-27 12:58:43 +01:00
}
2021-05-13 08:15:14 +02:00
im.mboxStatus = status
2021-01-28 09:40:08 +01:00
return nil
}
2021-05-13 08:15:14 +02:00
func (im *IMAP) Messages(folder string) ([]*Message, error) {
2021-05-15 11:19:28 +02:00
if err := im.Connect(); err != nil {
return []*Message{}, err
}
2021-05-13 08:15:14 +02:00
defer im.Close()
if err := im.selectFolder(folder); err != nil {
2021-01-29 12:29:23 +01:00
return []*Message{}, err
2021-01-28 09:40:08 +01:00
}
2021-01-27 12:58:43 +01:00
2021-05-13 08:15:14 +02:00
if im.mboxStatus.Messages == 0 {
2021-01-28 14:43:00 +01:00
return []*Message{}, nil
}
2021-01-27 12:58:43 +01:00
seqset := new(imap.SeqSet)
2021-05-13 08:15:14 +02:00
seqset.AddRange(uint32(1), im.mboxStatus.Messages)
2021-01-27 12:58:43 +01:00
2021-01-29 17:22:07 +01:00
// Get the whole message body
section := &imap.BodySectionName{}
items := []imap.FetchItem{imap.FetchUid, section.FetchItem()}
2021-01-27 12:58:43 +01:00
imsg, done := make(chan *imap.Message), make(chan error)
go func() {
2021-05-13 08:15:14 +02:00
done <- im.client.Fetch(seqset, items, imsg)
2021-01-27 12:58:43 +01:00
}()
messages := []*Message{}
for m := range imsg {
2021-01-29 17:22:07 +01:00
r := m.GetBody(section)
if r == nil {
return []*Message{}, fmt.Errorf("server didn't returned message body")
}
// Create a new mail reader
mr, err := mail.CreateReader(r)
if err != nil {
return []*Message{}, err
}
// Print some info about the message
header := mr.Header
subject, err := header.Subject()
if err != nil {
return []*Message{}, err
}
// Process each message's part
body := []byte(``)
for {
p, err := mr.NextPart()
if err == io.EOF {
break
} else if err != nil {
return []*Message{}, err
}
switch p.Header.(type) {
case *mail.InlineHeader:
// This is the message's text (can be plain-text or HTML)
body, _ = ioutil.ReadAll(p.Body)
}
}
2021-01-27 12:58:43 +01:00
messages = append(messages, &Message{
2021-01-28 14:43:00 +01:00
Uid: m.Uid,
2021-01-29 12:29:23 +01:00
Folder: folder,
2021-01-29 17:22:07 +01:00
Subject: subject,
Body: string(body),
2021-01-27 12:58:43 +01:00
})
}
if err := <-done; err != nil {
2021-05-13 08:15:14 +02:00
return []*Message{}, fmt.Errorf("%w: %v", ErrIMAPServerProblem, err)
2021-01-27 12:58:43 +01:00
}
2021-08-21 11:03:30 +02:00
// for reasons yet unknown, with some imap providers (i.e. Fastmail) the code
// above sometimes returns the same message twice, but with a different uid.
dedupMessages := []*Message{}
for _, m := range messages {
var isDupe bool
2021-08-21 11:03:30 +02:00
for _, dm := range dedupMessages {
if m.Equal(dm) {
isDupe = true
break
2021-08-21 11:03:30 +02:00
}
}
if !isDupe {
dedupMessages = append(dedupMessages, m)
}
2021-08-21 11:03:30 +02:00
}
return dedupMessages, nil
2021-01-27 12:58:43 +01:00
}
2021-01-27 15:01:42 +01:00
2021-05-13 08:15:14 +02:00
func (im *IMAP) Add(folder, subject, body string) error {
2021-05-15 11:19:28 +02:00
if err := im.Connect(); err != nil {
return err
}
2021-05-13 08:15:14 +02:00
defer im.Close()
2021-08-13 08:45:12 +02:00
msgStr := fmt.Sprintf("From: %s\r\nDate: %s\r\nSubject: %s\r\n\r\n%s\r\n",
2021-09-19 08:46:55 +02:00
"gte <gte@ewintr.nl>",
2021-08-13 08:45:12 +02:00
time.Now().Format(time.RFC822Z),
subject,
body,
)
2021-05-13 08:15:14 +02:00
msg := NewIMAPBody(msgStr)
2021-01-29 12:29:23 +01:00
folder = fmt.Sprintf("%s%s", im.config.IMAPFolderPrefix, folder)
2021-05-13 08:15:14 +02:00
if err := im.client.Append(folder, nil, time.Time{}, imap.Literal(msg)); err != nil {
return fmt.Errorf("%w: %v", ErrIMAPServerProblem, err)
}
return nil
2021-01-27 15:01:42 +01:00
}
2021-01-28 09:40:08 +01:00
2021-05-13 08:15:14 +02:00
func (im *IMAP) Remove(msg *Message) error {
2021-01-30 11:20:12 +01:00
if msg == nil || !msg.Valid() {
2021-01-29 12:29:23 +01:00
return ErrInvalidMessage
2021-01-28 14:43:00 +01:00
}
2021-05-15 11:19:28 +02:00
if err := im.Connect(); err != nil {
return err
}
2021-05-13 08:15:14 +02:00
defer im.Close()
2021-01-29 12:29:23 +01:00
2021-05-13 08:15:14 +02:00
if err := im.selectFolder(msg.Folder); err != nil {
2021-01-29 12:29:23 +01:00
return err
2021-01-28 09:40:08 +01:00
}
// set deleted flag
seqset := new(imap.SeqSet)
2021-01-29 12:29:23 +01:00
seqset.AddRange(msg.Uid, msg.Uid)
2021-01-28 09:40:08 +01:00
storeItem := imap.FormatFlagsOp(imap.SetFlags, true)
2021-05-13 08:15:14 +02:00
err := im.client.UidStore(seqset, storeItem, imap.FormatStringList([]string{imap.DeletedFlag}), nil)
2021-01-28 09:40:08 +01:00
if err != nil {
2021-05-13 08:15:14 +02:00
return fmt.Errorf("%w: %v", ErrIMAPServerProblem, err)
2021-01-28 09:40:08 +01:00
}
// expunge box
2021-05-13 08:15:14 +02:00
if err := im.client.Expunge(nil); err != nil {
return fmt.Errorf("%w: %v", ErrIMAPServerProblem, err)
2021-01-28 09:40:08 +01:00
}
return nil
}