mirror of
https://github.com/mjl-/mox.git
synced 2025-01-14 01:06:27 +03:00
40ade995a5
- add option to put messages in the queue "on hold", preventing delivery attempts until taken off hold again. - add "hold rules", to automatically mark some/all submitted messages as "on hold", e.g. from a specific account or to a specific domain. - add operation to "fail" a message, causing a DSN to be delivered to the sender. previously we could only drop a message from the queue. - update admin page & add new cli tools for these operations, with new filtering rules for selecting the messages to operate on. in the admin interface, add filtering and checkboxes to select a set of messages to operate on.
405 lines
10 KiB
Go
405 lines
10 KiB
Go
package main
|
|
|
|
import (
|
|
"encoding/json"
|
|
"flag"
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"os"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/mjl-/mox/queue"
|
|
)
|
|
|
|
func cmdQueueHoldrulesList(c *cmd) {
|
|
c.help = `List hold rules for the delivery queue.
|
|
|
|
Messages submitted to the queue that match a hold rule will be marked as on hold
|
|
and not scheduled for delivery.
|
|
`
|
|
if len(c.Parse()) != 0 {
|
|
c.Usage()
|
|
}
|
|
mustLoadConfig()
|
|
ctlcmdQueueHoldrulesList(xctl())
|
|
}
|
|
|
|
func ctlcmdQueueHoldrulesList(ctl *ctl) {
|
|
ctl.xwrite("queueholdruleslist")
|
|
ctl.xreadok()
|
|
if _, err := io.Copy(os.Stdout, ctl.reader()); err != nil {
|
|
log.Fatalf("%s", err)
|
|
}
|
|
}
|
|
|
|
func cmdQueueHoldrulesAdd(c *cmd) {
|
|
c.params = "[ruleflags]"
|
|
c.help = `Add hold rule for the delivery queue.
|
|
|
|
Add a hold rule to mark matching newly submitted messages as on hold. Set the
|
|
matching rules with the flags. Don't specify any flags to match all submitted
|
|
messages.
|
|
`
|
|
var account, senderDomain, recipientDomain string
|
|
c.flag.StringVar(&account, "account", "", "account submitting the message")
|
|
c.flag.StringVar(&senderDomain, "senderdom", "", "sender domain")
|
|
c.flag.StringVar(&recipientDomain, "recipientdom", "", "recipient domain")
|
|
if len(c.Parse()) != 0 {
|
|
c.Usage()
|
|
}
|
|
mustLoadConfig()
|
|
ctlcmdQueueHoldrulesAdd(xctl(), account, senderDomain, recipientDomain)
|
|
}
|
|
|
|
func ctlcmdQueueHoldrulesAdd(ctl *ctl, account, senderDomain, recipientDomain string) {
|
|
ctl.xwrite("queueholdrulesadd")
|
|
ctl.xwrite(account)
|
|
ctl.xwrite(senderDomain)
|
|
ctl.xwrite(recipientDomain)
|
|
ctl.xreadok()
|
|
}
|
|
|
|
func cmdQueueHoldrulesRemove(c *cmd) {
|
|
c.params = "ruleid"
|
|
c.help = `Remove hold rule for the delivery queue.
|
|
|
|
Remove a hold rule by its id.
|
|
`
|
|
args := c.Parse()
|
|
if len(args) != 1 {
|
|
c.Usage()
|
|
}
|
|
id, err := strconv.ParseInt(args[0], 10, 64)
|
|
xcheckf(err, "parsing id")
|
|
mustLoadConfig()
|
|
ctlcmdQueueHoldrulesRemove(xctl(), id)
|
|
}
|
|
|
|
func ctlcmdQueueHoldrulesRemove(ctl *ctl, id int64) {
|
|
ctl.xwrite("queueholdrulesremove")
|
|
ctl.xwrite(fmt.Sprintf("%d", id))
|
|
ctl.xreadok()
|
|
}
|
|
|
|
// flagFilter is used by many of the queue commands to accept flags for filtering
|
|
// the messages the operation applies to.
|
|
func flagFilter(fs *flag.FlagSet, f *queue.Filter) {
|
|
fs.Func("ids", "comma-separated list of message IDs", func(v string) error {
|
|
for _, s := range strings.Split(v, ",") {
|
|
id, err := strconv.ParseInt(s, 10, 64)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
f.IDs = append(f.IDs, id)
|
|
}
|
|
return nil
|
|
})
|
|
fs.StringVar(&f.Account, "account", "", "account that queued the message")
|
|
fs.StringVar(&f.From, "from", "", `from address of message, use "@example.com" to match all messages for a domain`)
|
|
fs.StringVar(&f.To, "to", "", `recipient address of message, use "@example.com" to match all messages for a domain`)
|
|
fs.StringVar(&f.Submitted, "submitted", "", `filter by time of submission relative to now, value must start with "<" (before now) or ">" (after now)`)
|
|
fs.StringVar(&f.NextAttempt, "nextattempt", "", `filter by time of next delivery attempt relative to now, value must start with "<" (before now) or ">" (after now)`)
|
|
fs.Func("transport", "transport to use for messages, empty string sets the default behaviour", func(v string) error {
|
|
f.Transport = &v
|
|
return nil
|
|
})
|
|
fs.Func("hold", "true or false, whether to match only messages that are (not) on hold", func(v string) error {
|
|
var hold bool
|
|
if v == "true" {
|
|
hold = true
|
|
} else if v == "false" {
|
|
hold = false
|
|
} else {
|
|
return fmt.Errorf("bad value %q", v)
|
|
}
|
|
f.Hold = &hold
|
|
return nil
|
|
})
|
|
}
|
|
|
|
func cmdQueueList(c *cmd) {
|
|
c.params = "[filterflags]"
|
|
c.help = `List matching messages in the delivery queue.
|
|
|
|
Prints the message with its ID, last and next delivery attempts, last error.
|
|
`
|
|
var f queue.Filter
|
|
flagFilter(c.flag, &f)
|
|
if len(c.Parse()) != 0 {
|
|
c.Usage()
|
|
}
|
|
mustLoadConfig()
|
|
ctlcmdQueueList(xctl(), f)
|
|
}
|
|
|
|
func xctlwritequeuefilter(ctl *ctl, f queue.Filter) {
|
|
fbuf, err := json.Marshal(f)
|
|
xcheckf(err, "marshal filter")
|
|
ctl.xwrite(string(fbuf))
|
|
}
|
|
|
|
func ctlcmdQueueList(ctl *ctl, f queue.Filter) {
|
|
ctl.xwrite("queuelist")
|
|
xctlwritequeuefilter(ctl, f)
|
|
ctl.xreadok()
|
|
if _, err := io.Copy(os.Stdout, ctl.reader()); err != nil {
|
|
log.Fatalf("%s", err)
|
|
}
|
|
}
|
|
|
|
func cmdQueueHold(c *cmd) {
|
|
c.params = "[filterflags]"
|
|
c.help = `Mark matching messages on hold.
|
|
|
|
Messages that are on hold are not delivered until marked as off hold again, or
|
|
otherwise handled by the admin.
|
|
`
|
|
var f queue.Filter
|
|
flagFilter(c.flag, &f)
|
|
if len(c.Parse()) != 0 {
|
|
c.Usage()
|
|
}
|
|
mustLoadConfig()
|
|
ctlcmdQueueHoldSet(xctl(), f, true)
|
|
}
|
|
|
|
func cmdQueueUnhold(c *cmd) {
|
|
c.params = "[filterflags]"
|
|
c.help = `Mark matching messages off hold.
|
|
|
|
Once off hold, messages can be delivered according to their current next
|
|
delivery attempt. See the "queue schedule" command.
|
|
`
|
|
var f queue.Filter
|
|
flagFilter(c.flag, &f)
|
|
if len(c.Parse()) != 0 {
|
|
c.Usage()
|
|
}
|
|
mustLoadConfig()
|
|
ctlcmdQueueHoldSet(xctl(), f, false)
|
|
}
|
|
|
|
func ctlcmdQueueHoldSet(ctl *ctl, f queue.Filter, hold bool) {
|
|
ctl.xwrite("queueholdset")
|
|
xctlwritequeuefilter(ctl, f)
|
|
if hold {
|
|
ctl.xwrite("true")
|
|
} else {
|
|
ctl.xwrite("false")
|
|
}
|
|
line := ctl.xread()
|
|
if line == "ok" {
|
|
fmt.Printf("%s messages changed\n", ctl.xread())
|
|
} else {
|
|
log.Fatalf("%s", line)
|
|
}
|
|
}
|
|
|
|
func cmdQueueSchedule(c *cmd) {
|
|
c.params = "[filterflags] duration"
|
|
c.help = `Change next delivery attempt for matching messages.
|
|
|
|
The next delivery attempt is adjusted by the duration parameter. If the -now
|
|
flag is set, the new delivery attempt is set to the duration added to the
|
|
current time, instead of added to the current scheduled time.
|
|
|
|
Schedule immediate delivery with "mox queue schedule -now 0".
|
|
`
|
|
var fromNow bool
|
|
c.flag.BoolVar(&fromNow, "now", false, "schedule for duration relative to current time instead of relative to current next delivery attempt for messages")
|
|
var f queue.Filter
|
|
flagFilter(c.flag, &f)
|
|
args := c.Parse()
|
|
if len(args) != 1 {
|
|
c.Usage()
|
|
}
|
|
d, err := time.ParseDuration(args[0])
|
|
xcheckf(err, "parsing duration %q", args[0])
|
|
mustLoadConfig()
|
|
ctlcmdQueueSchedule(xctl(), f, fromNow, d)
|
|
}
|
|
|
|
func ctlcmdQueueSchedule(ctl *ctl, f queue.Filter, fromNow bool, d time.Duration) {
|
|
ctl.xwrite("queueschedule")
|
|
xctlwritequeuefilter(ctl, f)
|
|
if fromNow {
|
|
ctl.xwrite("yes")
|
|
} else {
|
|
ctl.xwrite("")
|
|
}
|
|
ctl.xwrite(d.String())
|
|
line := ctl.xread()
|
|
if line == "ok" {
|
|
fmt.Printf("%s messages rescheduled\n", ctl.xread())
|
|
} else {
|
|
log.Fatalf("%s", line)
|
|
}
|
|
}
|
|
|
|
func cmdQueueTransport(c *cmd) {
|
|
c.params = "[filterflags] transport"
|
|
c.help = `Set transport for matching messages.
|
|
|
|
By default, the routing rules determine how a message is delivered. The default
|
|
and common case is direct delivery with SMTP. Messages can get a previously
|
|
configured transport assigned to use for delivery, e.g. using submission to
|
|
another mail server or with connections over a SOCKS proxy.
|
|
`
|
|
var f queue.Filter
|
|
flagFilter(c.flag, &f)
|
|
args := c.Parse()
|
|
if len(args) != 1 {
|
|
c.Usage()
|
|
}
|
|
mustLoadConfig()
|
|
ctlcmdQueueTransport(xctl(), f, args[0])
|
|
}
|
|
|
|
func ctlcmdQueueTransport(ctl *ctl, f queue.Filter, transport string) {
|
|
ctl.xwrite("queuetransport")
|
|
xctlwritequeuefilter(ctl, f)
|
|
ctl.xwrite(transport)
|
|
line := ctl.xread()
|
|
if line == "ok" {
|
|
fmt.Printf("%s messages changed\n", ctl.xread())
|
|
} else {
|
|
log.Fatalf("%s", line)
|
|
}
|
|
}
|
|
|
|
func cmdQueueRequireTLS(c *cmd) {
|
|
c.params = "[filterflags] {yes | no | default}"
|
|
c.help = `Set TLS requirements for delivery of matching messages.
|
|
|
|
Value "yes" is handled as if the RequireTLS extension was specified during
|
|
submission.
|
|
|
|
Value "no" is handled as if the message has a header "TLS-Required: No". This
|
|
header is not added by the queue. If messages without this header are relayed
|
|
through other mail servers they will apply their own default TLS policy.
|
|
|
|
Value "default" is the default behaviour, currently for unverified opportunistic
|
|
TLS.
|
|
`
|
|
var f queue.Filter
|
|
flagFilter(c.flag, &f)
|
|
args := c.Parse()
|
|
if len(args) != 1 {
|
|
c.Usage()
|
|
}
|
|
var tlsreq *bool
|
|
switch args[0] {
|
|
case "yes":
|
|
v := true
|
|
tlsreq = &v
|
|
case "no":
|
|
v := false
|
|
tlsreq = &v
|
|
case "default":
|
|
default:
|
|
c.Usage()
|
|
}
|
|
mustLoadConfig()
|
|
ctlcmdQueueRequireTLS(xctl(), f, tlsreq)
|
|
}
|
|
|
|
func ctlcmdQueueRequireTLS(ctl *ctl, f queue.Filter, tlsreq *bool) {
|
|
ctl.xwrite("queuerequiretls")
|
|
xctlwritequeuefilter(ctl, f)
|
|
var req string
|
|
if tlsreq == nil {
|
|
req = ""
|
|
} else if *tlsreq {
|
|
req = "true"
|
|
} else {
|
|
req = "false"
|
|
}
|
|
ctl.xwrite(req)
|
|
line := ctl.xread()
|
|
if line == "ok" {
|
|
fmt.Printf("%s messages changed\n", ctl.xread())
|
|
} else {
|
|
log.Fatalf("%s", line)
|
|
}
|
|
}
|
|
|
|
func cmdQueueFail(c *cmd) {
|
|
c.params = "[filterflags]"
|
|
c.help = `Fail delivery of matching messages, delivering DSNs.
|
|
|
|
Failing a message is handled similar to how delivery is given up after all
|
|
delivery attempts failed. The DSN (delivery status notification) message
|
|
contains a line saying the message was canceled by the admin.
|
|
`
|
|
var f queue.Filter
|
|
flagFilter(c.flag, &f)
|
|
if len(c.Parse()) != 0 {
|
|
c.Usage()
|
|
}
|
|
mustLoadConfig()
|
|
ctlcmdQueueFail(xctl(), f)
|
|
}
|
|
|
|
func ctlcmdQueueFail(ctl *ctl, f queue.Filter) {
|
|
ctl.xwrite("queuefail")
|
|
xctlwritequeuefilter(ctl, f)
|
|
line := ctl.xread()
|
|
if line == "ok" {
|
|
fmt.Printf("%s messages marked as failed\n", ctl.xread())
|
|
} else {
|
|
log.Fatalf("%s", line)
|
|
}
|
|
}
|
|
|
|
func cmdQueueDrop(c *cmd) {
|
|
c.params = "[filterflags]"
|
|
c.help = `Remove matching messages from the queue.
|
|
|
|
Dangerous operation, this completely removes the message. If you want to store
|
|
the message, use "queue dump" before removing.
|
|
`
|
|
var f queue.Filter
|
|
flagFilter(c.flag, &f)
|
|
if len(c.Parse()) != 0 {
|
|
c.Usage()
|
|
}
|
|
mustLoadConfig()
|
|
ctlcmdQueueDrop(xctl(), f)
|
|
}
|
|
|
|
func ctlcmdQueueDrop(ctl *ctl, f queue.Filter) {
|
|
ctl.xwrite("queuedrop")
|
|
xctlwritequeuefilter(ctl, f)
|
|
line := ctl.xread()
|
|
if line == "ok" {
|
|
fmt.Printf("%s messages dropped\n", ctl.xread())
|
|
} else {
|
|
log.Fatalf("%s", line)
|
|
}
|
|
}
|
|
|
|
func cmdQueueDump(c *cmd) {
|
|
c.params = "id"
|
|
c.help = `Dump a message from the queue.
|
|
|
|
The message is printed to stdout and is in standard internet mail format.
|
|
`
|
|
args := c.Parse()
|
|
if len(args) != 1 {
|
|
c.Usage()
|
|
}
|
|
mustLoadConfig()
|
|
ctlcmdQueueDump(xctl(), args[0])
|
|
}
|
|
|
|
func ctlcmdQueueDump(ctl *ctl, id string) {
|
|
ctl.xwrite("queuedump")
|
|
ctl.xwrite(id)
|
|
ctl.xreadok()
|
|
if _, err := io.Copy(os.Stdout, ctl.reader()); err != nil {
|
|
log.Fatalf("%s", err)
|
|
}
|
|
}
|