forked from Silverfish/proton-bridge
We build too many walls and not enough bridges
This commit is contained in:
263
internal/store/mailbox_ids.go
Normal file
263
internal/store/mailbox_ids.go
Normal file
@ -0,0 +1,263 @@
|
||||
// Copyright (c) 2020 Proton Technologies AG
|
||||
//
|
||||
// This file is part of ProtonMail Bridge.
|
||||
//
|
||||
// ProtonMail Bridge is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU General Public License as published by
|
||||
// the Free Software Foundation, either version 3 of the License, or
|
||||
// (at your option) any later version.
|
||||
//
|
||||
// ProtonMail Bridge is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
// GNU General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with ProtonMail Bridge. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
package store
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"math"
|
||||
"net/mail"
|
||||
"regexp"
|
||||
"strings"
|
||||
|
||||
"github.com/ProtonMail/proton-bridge/internal/imap/uidplus"
|
||||
"github.com/pkg/errors"
|
||||
bolt "go.etcd.io/bbolt"
|
||||
)
|
||||
|
||||
// GetAPIIDsFromUIDRange returns API IDs by IMAP UID range.
|
||||
//
|
||||
// API IDs are the long base64 strings that the API uses to identify messages.
|
||||
// UIDs are unique increasing integers that must be unique within a mailbox.
|
||||
func (storeMailbox *Mailbox) GetAPIIDsFromUIDRange(start, stop uint32) (apiIDs []string, err error) {
|
||||
err = storeMailbox.db().View(func(tx *bolt.Tx) error {
|
||||
b := storeMailbox.txGetIMAPIDsBucket(tx)
|
||||
|
||||
if stop == 0 {
|
||||
// A null stop means no stop.
|
||||
stop = ^uint32(0)
|
||||
}
|
||||
|
||||
startb := itob(start)
|
||||
stopb := itob(stop)
|
||||
|
||||
c := b.Cursor()
|
||||
for k, v := c.Seek(startb); k != nil && bytes.Compare(k, stopb) <= 0; k, v = c.Next() {
|
||||
apiIDs = append(apiIDs, string(v))
|
||||
}
|
||||
|
||||
return nil
|
||||
})
|
||||
return
|
||||
}
|
||||
|
||||
// GetAPIIDsFromSequenceRange returns API IDs by IMAP sequence number range.
|
||||
func (storeMailbox *Mailbox) GetAPIIDsFromSequenceRange(start, stop uint32) (apiIDs []string, err error) {
|
||||
err = storeMailbox.db().View(func(tx *bolt.Tx) error {
|
||||
b := storeMailbox.txGetIMAPIDsBucket(tx)
|
||||
c := b.Cursor()
|
||||
var i uint32
|
||||
for k, v := c.First(); k != nil; k, v = c.Next() {
|
||||
i++
|
||||
if i < start {
|
||||
continue
|
||||
}
|
||||
if stop > 0 && i > stop {
|
||||
break
|
||||
}
|
||||
apiIDs = append(apiIDs, string(v))
|
||||
}
|
||||
return nil
|
||||
})
|
||||
return
|
||||
}
|
||||
|
||||
// GetLatestAPIID returns the latest message API ID which still exists.
|
||||
// Info: not the latest IMAP UID which can be already removed.
|
||||
func (storeMailbox *Mailbox) GetLatestAPIID() (apiID string, err error) {
|
||||
err = storeMailbox.db().View(func(tx *bolt.Tx) error {
|
||||
b := storeMailbox.txGetAPIIDsBucket(tx)
|
||||
c := b.Cursor()
|
||||
lastAPIID, _ := c.Last()
|
||||
apiID = string(lastAPIID)
|
||||
if apiID == "" {
|
||||
return errors.New("cannot get latest API ID: empty mailbox")
|
||||
}
|
||||
return nil
|
||||
})
|
||||
return
|
||||
}
|
||||
|
||||
// GetNextUID returns the next IMAP UID.
|
||||
func (storeMailbox *Mailbox) GetNextUID() (uid uint32, err error) {
|
||||
err = storeMailbox.db().View(func(tx *bolt.Tx) error {
|
||||
b := storeMailbox.txGetIMAPIDsBucket(tx)
|
||||
uid, err = storeMailbox.txGetNextUID(b, false)
|
||||
return err
|
||||
})
|
||||
return
|
||||
}
|
||||
|
||||
func (storeMailbox *Mailbox) txGetNextUID(imapIDBucket *bolt.Bucket, write bool) (uint32, error) {
|
||||
var uid uint64
|
||||
var err error
|
||||
if write {
|
||||
uid, err = imapIDBucket.NextSequence()
|
||||
if err != nil {
|
||||
return 0, err
|
||||
}
|
||||
} else {
|
||||
uid = imapIDBucket.Sequence() + 1
|
||||
}
|
||||
if math.MaxUint32 <= uid {
|
||||
return 0, errors.New("too large sequence number")
|
||||
}
|
||||
return uint32(uid), nil
|
||||
}
|
||||
|
||||
// getUID returns IMAP UID in this mailbox for message ID.
|
||||
func (storeMailbox *Mailbox) getUID(apiID string) (uid uint32, err error) {
|
||||
err = storeMailbox.db().View(func(tx *bolt.Tx) error {
|
||||
uid, err = storeMailbox.txGetUID(tx, apiID)
|
||||
return err
|
||||
})
|
||||
return
|
||||
}
|
||||
|
||||
func (storeMailbox *Mailbox) txGetUID(tx *bolt.Tx, apiID string) (uint32, error) {
|
||||
b := storeMailbox.txGetAPIIDsBucket(tx)
|
||||
v := b.Get([]byte(apiID))
|
||||
if v == nil {
|
||||
return 0, ErrNoSuchAPIID
|
||||
}
|
||||
return btoi(v), nil
|
||||
}
|
||||
|
||||
// getSequenceNumber returns IMAP sequence number in the mailbox for the message with the given API ID `apiID`.
|
||||
func (storeMailbox *Mailbox) getSequenceNumber(apiID string) (seqNum uint32, err error) {
|
||||
err = storeMailbox.db().View(func(tx *bolt.Tx) error {
|
||||
b := storeMailbox.txGetIMAPIDsBucket(tx)
|
||||
uid, err := storeMailbox.txGetUID(tx, apiID)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
seqNum, err = storeMailbox.txGetSequenceNumberOfUID(b, itob(uid))
|
||||
return err
|
||||
})
|
||||
return
|
||||
}
|
||||
|
||||
// txGetSequenceNumberOfUID returns the IMAP sequence number of the message
|
||||
// with the given IMAP UID bytes `uidb`.
|
||||
//
|
||||
// NOTE: The `bolt.Cursor.Next()` loops in order of ascending key bytes. The
|
||||
// IMAP UID bucket is ordered by increasing UID because it's using BigEndian to
|
||||
// encode uint into byte. Hence the sequence number (IMAP ID) corresponds to
|
||||
// position of uid key in this order.
|
||||
func (storeMailbox *Mailbox) txGetSequenceNumberOfUID(bucket *bolt.Bucket, uidb []byte) (uint32, error) {
|
||||
seqNum := uint32(0)
|
||||
c := bucket.Cursor()
|
||||
|
||||
// Speed up for the case of last message. This is always true for
|
||||
// adding new message. It will return number of keys in bucket because
|
||||
// sequence number starts with 1.
|
||||
// We cannot use bucket.Stats() for that--it doesn't work in the same
|
||||
// transaction because stats are updated when transaction is committed.
|
||||
// But we can at least optimise to not do equal for all keys.
|
||||
lastKey, _ := c.Last()
|
||||
isLast := bytes.Equal(lastKey, uidb)
|
||||
|
||||
for k, _ := c.First(); k != nil; k, _ = c.Next() {
|
||||
seqNum++ // Sequence number starts at 1.
|
||||
if isLast {
|
||||
continue
|
||||
}
|
||||
if bytes.Equal(k, uidb) {
|
||||
return seqNum, nil
|
||||
}
|
||||
}
|
||||
|
||||
if isLast {
|
||||
return seqNum, nil
|
||||
}
|
||||
|
||||
return 0, ErrNoSuchUID
|
||||
}
|
||||
|
||||
// GetUIDList returns UID list corresponding to messageIDs in a requested order.
|
||||
func (storeMailbox *Mailbox) GetUIDList(apiIDs []string) *uidplus.OrderedSeq {
|
||||
seqSet := &uidplus.OrderedSeq{}
|
||||
_ = storeMailbox.db().View(func(tx *bolt.Tx) error {
|
||||
b := storeMailbox.txGetAPIIDsBucket(tx)
|
||||
for _, apiID := range apiIDs {
|
||||
v := b.Get([]byte(apiID))
|
||||
if v == nil {
|
||||
storeMailbox.log.
|
||||
WithField("msgID", apiID).
|
||||
Warn("Cannot find UID")
|
||||
continue
|
||||
}
|
||||
|
||||
seqSet.Add(btoi(v))
|
||||
}
|
||||
return nil
|
||||
})
|
||||
return seqSet
|
||||
}
|
||||
|
||||
// GetUIDByHeader returns UID of message existing in mailbox or zero if no match found.
|
||||
func (storeMailbox *Mailbox) GetUIDByHeader(header *mail.Header) (foundUID uint32) {
|
||||
if header == nil {
|
||||
return uint32(0)
|
||||
}
|
||||
|
||||
// Message-Id in appended-after-send mail is processed as ExternalID
|
||||
// in PM message. Message-Id in normal copy/move will be the PM internal ID.
|
||||
messageID := header.Get("Message-Id")
|
||||
|
||||
// The most often situation is that message is APPENDed after it was sent so the
|
||||
// Message-ID will be reflected by ExternalID in API message meta-data.
|
||||
externalID := strings.Trim(messageID, "<> ") // remove '<>' to improve match
|
||||
matchExternalID := regexp.MustCompile(`"ExternalID":"` +
|
||||
` *(\\u003c)? *` + // \u003c is equivalent to `<`
|
||||
regexp.QuoteMeta(externalID) +
|
||||
` *(\\u003e)? *` + // \u0033 is equivalent to `>`
|
||||
`"`,
|
||||
)
|
||||
|
||||
// It is possible that client will try to COPY existing message to Sent
|
||||
// using APPEND command. In that case the Message-Id from header will
|
||||
// be internal message ID and we need to check whether it's already there.
|
||||
matchInternalID := bytes.Split([]byte(externalID), []byte("@"))[0]
|
||||
|
||||
_ = storeMailbox.db().View(func(tx *bolt.Tx) error {
|
||||
metaBucket := tx.Bucket(metadataBucket)
|
||||
b := storeMailbox.txGetIMAPIDsBucket(tx)
|
||||
c := b.Cursor()
|
||||
imapID, apiID := c.Last()
|
||||
for ; imapID != nil; imapID, apiID = c.Prev() {
|
||||
rawMeta := metaBucket.Get(apiID)
|
||||
if rawMeta == nil {
|
||||
storeMailbox.log.
|
||||
WithField("IMAP-UID", imapID).
|
||||
WithField("API-ID", apiID).
|
||||
Warn("Cannot find meta-data while searching for externalID")
|
||||
continue
|
||||
}
|
||||
|
||||
if !matchExternalID.Match(rawMeta) && !bytes.Equal(apiID, matchInternalID) {
|
||||
continue
|
||||
}
|
||||
|
||||
foundUID = btoi(imapID)
|
||||
return nil
|
||||
}
|
||||
return nil
|
||||
})
|
||||
|
||||
return foundUID
|
||||
}
|
||||
Reference in New Issue
Block a user