Use Unix socket to pass IMAP updates from maddyctl to daemon

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.
This commit is contained in:
fox.cpp 2019-12-08 03:27:31 +03:00
parent 9e5bb288b3
commit a574b9fbb2
No known key found for this signature in database
GPG key ID: E76D97CCEDE90B6C
10 changed files with 431 additions and 11 deletions

View file

@ -0,0 +1,112 @@
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
}