mirror of
https://github.com/foxcpp/maddy.git
synced 2025-04-04 21:47:40 +03:00
There is abstraction 'updates pipe' defined for future use with configuration involving IMAP data replication (e.g. multiple nodes with maddy instances + PostgreSQL replicas + S3 bucket for messages). However, for the case of local SQLite3 DB, limited UDS-based implementation is provided. It solves the problem of maddyctl not being able to tell the server about modifications it makes. Alternative to this approach would be to have server actually perform operations and maddyctl being a dumb API client, but this requires a lot more complex IPC interface and will not work when the server is down.
112 lines
2.6 KiB
Go
112 lines
2.6 KiB
Go
package updatepipe
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/emersion/go-imap"
|
|
"github.com/emersion/go-imap/backend"
|
|
)
|
|
|
|
func unescapeName(s string) string {
|
|
return strings.ReplaceAll(s, "\x10", ";")
|
|
}
|
|
|
|
func escapeName(s string) string {
|
|
return strings.ReplaceAll(s, ";", "\x10")
|
|
}
|
|
|
|
type message struct {
|
|
SeqNum uint32
|
|
Flags []string
|
|
}
|
|
|
|
func parseUpdate(s string) (id string, upd backend.Update, err error) {
|
|
parts := strings.SplitN(s, ";", 5)
|
|
if len(parts) != 5 {
|
|
return "", nil, errors.New("updatepipe: mismatched parts count")
|
|
}
|
|
|
|
updBase := backend.NewUpdate(unescapeName(parts[2]), unescapeName(parts[3]))
|
|
switch parts[1] {
|
|
case "ExpungeUpdate":
|
|
exUpd := &backend.ExpungeUpdate{Update: updBase}
|
|
if err := json.Unmarshal([]byte(parts[4]), &exUpd.SeqNum); err != nil {
|
|
return "", nil, err
|
|
}
|
|
upd = exUpd
|
|
case "MailboxUpdate":
|
|
mboxUpd := &backend.MailboxUpdate{Update: updBase}
|
|
if err := json.Unmarshal([]byte(parts[4]), &mboxUpd.MailboxStatus); err != nil {
|
|
return "", nil, err
|
|
}
|
|
upd = mboxUpd
|
|
case "MessageUpdate":
|
|
// imap.Message is not JSON-serializable because it contains maps with
|
|
// complex keys.
|
|
// In practice, however, MessageUpdate is used only for FLAGS, so we
|
|
// serialize them only with a SeqNum.
|
|
|
|
msg := message{}
|
|
if err := json.Unmarshal([]byte(parts[4]), &msg); err != nil {
|
|
return "", nil, err
|
|
}
|
|
|
|
msgUpd := &backend.MessageUpdate{
|
|
Update: updBase,
|
|
Message: imap.NewMessage(msg.SeqNum, []imap.FetchItem{imap.FetchFlags}),
|
|
}
|
|
msgUpd.Message.Flags = msg.Flags
|
|
upd = msgUpd
|
|
}
|
|
|
|
return parts[0], upd, nil
|
|
}
|
|
|
|
func formatUpdate(myID string, upd backend.Update) (string, error) {
|
|
var (
|
|
objType string
|
|
objStr []byte
|
|
err error
|
|
)
|
|
switch v := upd.(type) {
|
|
case *backend.ExpungeUpdate:
|
|
objType = "ExpungeUpdate"
|
|
objStr, err = json.Marshal(v.SeqNum)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
case *backend.MessageUpdate:
|
|
// imap.Message is not JSON-serializable because it contains maps with
|
|
// complex keys.
|
|
// In practice, however, MessageUpdate is used only for FLAGS, so we
|
|
// serialize them only with a seqnum.
|
|
|
|
objType = "MessageUpdate"
|
|
objStr, err = json.Marshal(message{
|
|
SeqNum: v.Message.SeqNum,
|
|
Flags: v.Message.Flags,
|
|
})
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
case *backend.MailboxUpdate:
|
|
objType = "MailboxUpdate"
|
|
objStr, err = json.Marshal(v.MailboxStatus)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
default:
|
|
return "", fmt.Errorf("updatepipe: unknown update type: %T", upd)
|
|
}
|
|
|
|
return strings.Join([]string{
|
|
myID,
|
|
objType,
|
|
escapeName(upd.Username()),
|
|
escapeName(upd.Mailbox()),
|
|
string(objStr),
|
|
}, ";") + "\n", nil
|
|
}
|