mirror of
https://github.com/mjl-/mox.git
synced 2025-01-28 07:15:55 +03:00
make some maintenance commands that were previously unlisted listed
we refer to these commands in output of "mox verifydata", so they should be findable other than through the code...
This commit is contained in:
parent
7cceb3d834
commit
a4c6fe815f
2 changed files with 240 additions and 144 deletions
92
doc.go
92
doc.go
|
@ -68,6 +68,14 @@ low-maintenance self-hosted email.
|
||||||
mox tlsrpt lookup domain
|
mox tlsrpt lookup domain
|
||||||
mox tlsrpt parsereportmsg message ...
|
mox tlsrpt parsereportmsg message ...
|
||||||
mox version
|
mox version
|
||||||
|
mox bumpuidvalidity account [mailbox]
|
||||||
|
mox reassignuids account [mailboxid]
|
||||||
|
mox fixuidmeta account
|
||||||
|
mox fixmsgsize [account]
|
||||||
|
mox reparse [account]
|
||||||
|
mox ensureparsed account
|
||||||
|
mox recalculatemailboxcounts account
|
||||||
|
mox message parse message.eml
|
||||||
|
|
||||||
Many commands talk to a running mox instance, through the ctl file in the data
|
Many commands talk to a running mox instance, through the ctl file in the data
|
||||||
directory. Specify the configuration file (that holds the path to the data
|
directory. Specify the configuration file (that holds the path to the data
|
||||||
|
@ -775,6 +783,90 @@ The report is printed in formatted JSON.
|
||||||
Prints this mox version.
|
Prints this mox version.
|
||||||
|
|
||||||
usage: mox version
|
usage: mox version
|
||||||
|
|
||||||
|
# mox bumpuidvalidity
|
||||||
|
|
||||||
|
Change the IMAP UID validity of the mailbox, causing IMAP clients to refetch messages.
|
||||||
|
|
||||||
|
This can be useful after manually repairing metadata about the account/mailbox.
|
||||||
|
|
||||||
|
Opens account database file directly. Ensure mox does not have the account
|
||||||
|
+open, or is not running.
|
||||||
|
|
||||||
|
usage: mox bumpuidvalidity account [mailbox]
|
||||||
|
|
||||||
|
# mox reassignuids
|
||||||
|
|
||||||
|
Reassign UIDs in one mailbox or all mailboxes in an account and bump UID validity, causing IMAP clients to refetch messages.
|
||||||
|
|
||||||
|
Opens account database file directly. Ensure mox does not have the account
|
||||||
|
open, or is not running.
|
||||||
|
|
||||||
|
usage: mox reassignuids account [mailboxid]
|
||||||
|
|
||||||
|
# mox fixuidmeta
|
||||||
|
|
||||||
|
Fix inconsistent UIDVALIDITY and UIDNEXT in messages/mailboxes/account.
|
||||||
|
|
||||||
|
The next UID to use for a message in a mailbox should always be higher than any
|
||||||
|
existing message UID in the mailbox. If it is not, the mailbox UIDNEXT is
|
||||||
|
updated.
|
||||||
|
|
||||||
|
Each mailbox has a UIDVALIDITY sequence number, which should always be lower
|
||||||
|
than the per-account next UIDVALIDITY to use. If it is not, the account next
|
||||||
|
UIDVALIDITY is updated.
|
||||||
|
|
||||||
|
Opens account database file directly. Ensure mox does not have the account
|
||||||
|
open, or is not running.
|
||||||
|
|
||||||
|
usage: mox fixuidmeta account
|
||||||
|
|
||||||
|
# mox fixmsgsize
|
||||||
|
|
||||||
|
Ensure message sizes in the database matching the sum of the message prefix length and on-disk file size.
|
||||||
|
|
||||||
|
Messages with an inconsistent size are also parsed again.
|
||||||
|
|
||||||
|
If an inconsistency is found, you should probably also run "mox
|
||||||
|
bumpuidvalidity" on the mailboxes or entire account to force IMAP clients to
|
||||||
|
refetch messages.
|
||||||
|
|
||||||
|
usage: mox fixmsgsize [account]
|
||||||
|
|
||||||
|
# mox reparse
|
||||||
|
|
||||||
|
# Parse all messages in the account or all accounts again
|
||||||
|
|
||||||
|
Can be useful after upgrading mox with improved message parsing. Messages are
|
||||||
|
parsed in batches, so other access to the mailboxes/messages are not blocked
|
||||||
|
while reparsing all messages.
|
||||||
|
|
||||||
|
usage: mox reparse [account]
|
||||||
|
|
||||||
|
# mox ensureparsed
|
||||||
|
|
||||||
|
Ensure messages in the database have a pre-parsed MIME form in the database.
|
||||||
|
|
||||||
|
usage: mox ensureparsed account
|
||||||
|
-all
|
||||||
|
store new parsed message for all messages
|
||||||
|
|
||||||
|
# mox recalculatemailboxcounts
|
||||||
|
|
||||||
|
Recalculate message counts for all mailboxes in the account.
|
||||||
|
|
||||||
|
When a message is added to/removed from a mailbox, or when message flags change,
|
||||||
|
the total, unread, unseen and deleted messages are accounted, and the total size
|
||||||
|
of the mailbox. In case of a bug in this accounting, the numbers could become
|
||||||
|
incorrect. This command will find, fix and print them.
|
||||||
|
|
||||||
|
usage: mox recalculatemailboxcounts account
|
||||||
|
|
||||||
|
# mox message parse
|
||||||
|
|
||||||
|
Parse message, print JSON representation.
|
||||||
|
|
||||||
|
usage: mox message parse message.eml
|
||||||
*/
|
*/
|
||||||
package main
|
package main
|
||||||
|
|
||||||
|
|
292
main.go
292
main.go
|
@ -132,6 +132,15 @@ var commands = []struct {
|
||||||
{"tlsrpt parsereportmsg", cmdTLSRPTParsereportmsg},
|
{"tlsrpt parsereportmsg", cmdTLSRPTParsereportmsg},
|
||||||
{"version", cmdVersion},
|
{"version", cmdVersion},
|
||||||
|
|
||||||
|
{"bumpuidvalidity", cmdBumpUIDValidity},
|
||||||
|
{"reassignuids", cmdReassignUIDs},
|
||||||
|
{"fixuidmeta", cmdFixUIDMeta},
|
||||||
|
{"fixmsgsize", cmdFixmsgsize},
|
||||||
|
{"reparse", cmdReparse},
|
||||||
|
{"ensureparsed", cmdEnsureParsed},
|
||||||
|
{"recalculatemailboxcounts", cmdRecalculateMailboxCounts},
|
||||||
|
{"message parse", cmdMessageParse},
|
||||||
|
|
||||||
// Not listed.
|
// Not listed.
|
||||||
{"helpall", cmdHelpall},
|
{"helpall", cmdHelpall},
|
||||||
{"junk analyze", cmdJunkAnalyze},
|
{"junk analyze", cmdJunkAnalyze},
|
||||||
|
@ -139,14 +148,7 @@ var commands = []struct {
|
||||||
{"junk play", cmdJunkPlay},
|
{"junk play", cmdJunkPlay},
|
||||||
{"junk test", cmdJunkTest},
|
{"junk test", cmdJunkTest},
|
||||||
{"junk train", cmdJunkTrain},
|
{"junk train", cmdJunkTrain},
|
||||||
{"bumpuidvalidity", cmdBumpUIDValidity},
|
|
||||||
{"reassignuids", cmdReassignUIDs},
|
|
||||||
{"fixuidmeta", cmdFixUIDMeta},
|
|
||||||
{"dmarcdb addreport", cmdDMARCDBAddReport},
|
{"dmarcdb addreport", cmdDMARCDBAddReport},
|
||||||
{"fixmsgsize", cmdFixmsgsize},
|
|
||||||
{"reparse", cmdReparse},
|
|
||||||
{"ensureparsed", cmdEnsureParsed},
|
|
||||||
{"message parse", cmdMessageParse},
|
|
||||||
{"tlsrptdb addreport", cmdTLSRPTDBAddReport},
|
{"tlsrptdb addreport", cmdTLSRPTDBAddReport},
|
||||||
{"updates addsigned", cmdUpdatesAddSigned},
|
{"updates addsigned", cmdUpdatesAddSigned},
|
||||||
{"updates genkey", cmdUpdatesGenkey},
|
{"updates genkey", cmdUpdatesGenkey},
|
||||||
|
@ -156,7 +158,6 @@ var commands = []struct {
|
||||||
{"gentestdata", cmdGentestdata},
|
{"gentestdata", cmdGentestdata},
|
||||||
{"ximport maildir", cmdXImportMaildir},
|
{"ximport maildir", cmdXImportMaildir},
|
||||||
{"ximport mbox", cmdXImportMbox},
|
{"ximport mbox", cmdXImportMbox},
|
||||||
{"recalculatemailboxcounts", cmdRecalculateMailboxCounts},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var cmds []cmd
|
var cmds []cmd
|
||||||
|
@ -1989,141 +1990,15 @@ func cmdVersion(c *cmd) {
|
||||||
fmt.Println(moxvar.Version)
|
fmt.Println(moxvar.Version)
|
||||||
}
|
}
|
||||||
|
|
||||||
func cmdFixmsgsize(c *cmd) {
|
|
||||||
c.unlisted = true
|
|
||||||
c.params = "[account]"
|
|
||||||
c.help = `Ensure message sizes in the database matching the sum of the message prefix length and on-disk file size.
|
|
||||||
|
|
||||||
Messages with an inconsistent size are also parsed again.
|
|
||||||
|
|
||||||
If an inconsistency is found, you should probably also run "mox
|
|
||||||
bumpuidvalidity" on the mailboxes or entire account to force IMAP clients to
|
|
||||||
refetch messages.
|
|
||||||
`
|
|
||||||
args := c.Parse()
|
|
||||||
if len(args) > 1 {
|
|
||||||
c.Usage()
|
|
||||||
}
|
|
||||||
|
|
||||||
mustLoadConfig()
|
|
||||||
var account string
|
|
||||||
if len(args) == 1 {
|
|
||||||
account = args[0]
|
|
||||||
}
|
|
||||||
ctlcmdFixmsgsize(xctl(), account)
|
|
||||||
}
|
|
||||||
|
|
||||||
func ctlcmdFixmsgsize(ctl *ctl, account string) {
|
|
||||||
ctl.xwrite("fixmsgsize")
|
|
||||||
ctl.xwrite(account)
|
|
||||||
ctl.xreadok()
|
|
||||||
ctl.xstreamto(os.Stdout)
|
|
||||||
}
|
|
||||||
|
|
||||||
func cmdReparse(c *cmd) {
|
|
||||||
c.unlisted = true
|
|
||||||
c.params = "[account]"
|
|
||||||
c.help = "Ensure messages in the database have a ParsedBuf."
|
|
||||||
args := c.Parse()
|
|
||||||
if len(args) > 1 {
|
|
||||||
c.Usage()
|
|
||||||
}
|
|
||||||
|
|
||||||
mustLoadConfig()
|
|
||||||
var account string
|
|
||||||
if len(args) == 1 {
|
|
||||||
account = args[0]
|
|
||||||
}
|
|
||||||
ctlcmdReparse(xctl(), account)
|
|
||||||
}
|
|
||||||
|
|
||||||
func ctlcmdReparse(ctl *ctl, account string) {
|
|
||||||
ctl.xwrite("reparse")
|
|
||||||
ctl.xwrite(account)
|
|
||||||
ctl.xreadok()
|
|
||||||
ctl.xstreamto(os.Stdout)
|
|
||||||
}
|
|
||||||
|
|
||||||
func cmdEnsureParsed(c *cmd) {
|
|
||||||
c.unlisted = true
|
|
||||||
c.params = "account"
|
|
||||||
c.help = "Ensure messages in the database have a ParsedBuf."
|
|
||||||
var all bool
|
|
||||||
c.flag.BoolVar(&all, "all", false, "store new parsed message for all messages")
|
|
||||||
args := c.Parse()
|
|
||||||
if len(args) != 1 {
|
|
||||||
c.Usage()
|
|
||||||
}
|
|
||||||
|
|
||||||
mustLoadConfig()
|
|
||||||
a, err := store.OpenAccount(args[0])
|
|
||||||
xcheckf(err, "open account")
|
|
||||||
defer func() {
|
|
||||||
if err := a.Close(); err != nil {
|
|
||||||
log.Printf("closing account: %v", err)
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
n := 0
|
|
||||||
err = a.DB.Write(context.Background(), func(tx *bstore.Tx) error {
|
|
||||||
q := bstore.QueryTx[store.Message](tx)
|
|
||||||
q.FilterEqual("Expunged", false)
|
|
||||||
q.FilterFn(func(m store.Message) bool {
|
|
||||||
return all || m.ParsedBuf == nil
|
|
||||||
})
|
|
||||||
l, err := q.List()
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("list messages: %v", err)
|
|
||||||
}
|
|
||||||
for _, m := range l {
|
|
||||||
mr := a.MessageReader(m)
|
|
||||||
p, err := message.EnsurePart(mr, m.Size)
|
|
||||||
if err != nil {
|
|
||||||
log.Printf("parsing message %d: %v (continuing)", m.ID, err)
|
|
||||||
}
|
|
||||||
m.ParsedBuf, err = json.Marshal(p)
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("marshal parsed message: %v", err)
|
|
||||||
}
|
|
||||||
if err := tx.Update(&m); err != nil {
|
|
||||||
return fmt.Errorf("update message: %v", err)
|
|
||||||
}
|
|
||||||
n++
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
})
|
|
||||||
xcheckf(err, "update messages with parsed mime structure")
|
|
||||||
fmt.Printf("%d messages updated\n", n)
|
|
||||||
}
|
|
||||||
|
|
||||||
func cmdMessageParse(c *cmd) {
|
|
||||||
c.unlisted = true
|
|
||||||
c.params = "message.eml"
|
|
||||||
c.help = "Parse message, print JSON representation."
|
|
||||||
|
|
||||||
args := c.Parse()
|
|
||||||
if len(args) != 1 {
|
|
||||||
c.Usage()
|
|
||||||
}
|
|
||||||
|
|
||||||
f, err := os.Open(args[0])
|
|
||||||
xcheckf(err, "open")
|
|
||||||
defer f.Close()
|
|
||||||
|
|
||||||
part, err := message.Parse(f)
|
|
||||||
xcheckf(err, "parsing message")
|
|
||||||
err = part.Walk(nil)
|
|
||||||
xcheckf(err, "parsing nested parts")
|
|
||||||
enc := json.NewEncoder(os.Stdout)
|
|
||||||
enc.SetIndent("", "\t")
|
|
||||||
err = enc.Encode(part)
|
|
||||||
xcheckf(err, "write")
|
|
||||||
}
|
|
||||||
|
|
||||||
func cmdBumpUIDValidity(c *cmd) {
|
func cmdBumpUIDValidity(c *cmd) {
|
||||||
c.unlisted = true
|
|
||||||
c.params = "account [mailbox]"
|
c.params = "account [mailbox]"
|
||||||
c.help = "Change the IMAP UID validity of the mailbox, causing IMAP clients to refetch messages."
|
c.help = `Change the IMAP UID validity of the mailbox, causing IMAP clients to refetch messages.
|
||||||
|
|
||||||
|
This can be useful after manually repairing metadata about the account/mailbox.
|
||||||
|
|
||||||
|
Opens account database file directly. Ensure mox does not have the account
|
||||||
|
+open, or is not running.
|
||||||
|
`
|
||||||
args := c.Parse()
|
args := c.Parse()
|
||||||
if len(args) != 1 && len(args) != 2 {
|
if len(args) != 1 && len(args) != 2 {
|
||||||
c.Usage()
|
c.Usage()
|
||||||
|
@ -2166,7 +2041,6 @@ func cmdBumpUIDValidity(c *cmd) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func cmdReassignUIDs(c *cmd) {
|
func cmdReassignUIDs(c *cmd) {
|
||||||
c.unlisted = true
|
|
||||||
c.params = "account [mailboxid]"
|
c.params = "account [mailboxid]"
|
||||||
c.help = `Reassign UIDs in one mailbox or all mailboxes in an account and bump UID validity, causing IMAP clients to refetch messages.
|
c.help = `Reassign UIDs in one mailbox or all mailboxes in an account and bump UID validity, causing IMAP clients to refetch messages.
|
||||||
|
|
||||||
|
@ -2260,7 +2134,6 @@ open, or is not running.
|
||||||
}
|
}
|
||||||
|
|
||||||
func cmdFixUIDMeta(c *cmd) {
|
func cmdFixUIDMeta(c *cmd) {
|
||||||
c.unlisted = true
|
|
||||||
c.params = "account"
|
c.params = "account"
|
||||||
c.help = `Fix inconsistent UIDVALIDITY and UIDNEXT in messages/mailboxes/account.
|
c.help = `Fix inconsistent UIDVALIDITY and UIDNEXT in messages/mailboxes/account.
|
||||||
|
|
||||||
|
@ -2333,8 +2206,116 @@ open, or is not running.
|
||||||
xcheckf(err, "updating database")
|
xcheckf(err, "updating database")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func cmdFixmsgsize(c *cmd) {
|
||||||
|
c.params = "[account]"
|
||||||
|
c.help = `Ensure message sizes in the database matching the sum of the message prefix length and on-disk file size.
|
||||||
|
|
||||||
|
Messages with an inconsistent size are also parsed again.
|
||||||
|
|
||||||
|
If an inconsistency is found, you should probably also run "mox
|
||||||
|
bumpuidvalidity" on the mailboxes or entire account to force IMAP clients to
|
||||||
|
refetch messages.
|
||||||
|
`
|
||||||
|
args := c.Parse()
|
||||||
|
if len(args) > 1 {
|
||||||
|
c.Usage()
|
||||||
|
}
|
||||||
|
|
||||||
|
mustLoadConfig()
|
||||||
|
var account string
|
||||||
|
if len(args) == 1 {
|
||||||
|
account = args[0]
|
||||||
|
}
|
||||||
|
ctlcmdFixmsgsize(xctl(), account)
|
||||||
|
}
|
||||||
|
|
||||||
|
func ctlcmdFixmsgsize(ctl *ctl, account string) {
|
||||||
|
ctl.xwrite("fixmsgsize")
|
||||||
|
ctl.xwrite(account)
|
||||||
|
ctl.xreadok()
|
||||||
|
ctl.xstreamto(os.Stdout)
|
||||||
|
}
|
||||||
|
|
||||||
|
func cmdReparse(c *cmd) {
|
||||||
|
c.params = "[account]"
|
||||||
|
c.help = `Parse all messages in the account or all accounts again
|
||||||
|
|
||||||
|
Can be useful after upgrading mox with improved message parsing. Messages are
|
||||||
|
parsed in batches, so other access to the mailboxes/messages are not blocked
|
||||||
|
while reparsing all messages.
|
||||||
|
`
|
||||||
|
args := c.Parse()
|
||||||
|
if len(args) > 1 {
|
||||||
|
c.Usage()
|
||||||
|
}
|
||||||
|
|
||||||
|
mustLoadConfig()
|
||||||
|
var account string
|
||||||
|
if len(args) == 1 {
|
||||||
|
account = args[0]
|
||||||
|
}
|
||||||
|
ctlcmdReparse(xctl(), account)
|
||||||
|
}
|
||||||
|
|
||||||
|
func ctlcmdReparse(ctl *ctl, account string) {
|
||||||
|
ctl.xwrite("reparse")
|
||||||
|
ctl.xwrite(account)
|
||||||
|
ctl.xreadok()
|
||||||
|
ctl.xstreamto(os.Stdout)
|
||||||
|
}
|
||||||
|
|
||||||
|
func cmdEnsureParsed(c *cmd) {
|
||||||
|
c.params = "account"
|
||||||
|
c.help = "Ensure messages in the database have a pre-parsed MIME form in the database."
|
||||||
|
var all bool
|
||||||
|
c.flag.BoolVar(&all, "all", false, "store new parsed message for all messages")
|
||||||
|
args := c.Parse()
|
||||||
|
if len(args) != 1 {
|
||||||
|
c.Usage()
|
||||||
|
}
|
||||||
|
|
||||||
|
mustLoadConfig()
|
||||||
|
a, err := store.OpenAccount(args[0])
|
||||||
|
xcheckf(err, "open account")
|
||||||
|
defer func() {
|
||||||
|
if err := a.Close(); err != nil {
|
||||||
|
log.Printf("closing account: %v", err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
n := 0
|
||||||
|
err = a.DB.Write(context.Background(), func(tx *bstore.Tx) error {
|
||||||
|
q := bstore.QueryTx[store.Message](tx)
|
||||||
|
q.FilterEqual("Expunged", false)
|
||||||
|
q.FilterFn(func(m store.Message) bool {
|
||||||
|
return all || m.ParsedBuf == nil
|
||||||
|
})
|
||||||
|
l, err := q.List()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("list messages: %v", err)
|
||||||
|
}
|
||||||
|
for _, m := range l {
|
||||||
|
mr := a.MessageReader(m)
|
||||||
|
p, err := message.EnsurePart(mr, m.Size)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("parsing message %d: %v (continuing)", m.ID, err)
|
||||||
|
}
|
||||||
|
m.ParsedBuf, err = json.Marshal(p)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("marshal parsed message: %v", err)
|
||||||
|
}
|
||||||
|
if err := tx.Update(&m); err != nil {
|
||||||
|
return fmt.Errorf("update message: %v", err)
|
||||||
|
}
|
||||||
|
n++
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
xcheckf(err, "update messages with parsed mime structure")
|
||||||
|
fmt.Printf("%d messages updated\n", n)
|
||||||
|
}
|
||||||
|
|
||||||
func cmdRecalculateMailboxCounts(c *cmd) {
|
func cmdRecalculateMailboxCounts(c *cmd) {
|
||||||
c.unlisted = true
|
|
||||||
c.params = "account"
|
c.params = "account"
|
||||||
c.help = `Recalculate message counts for all mailboxes in the account.
|
c.help = `Recalculate message counts for all mailboxes in the account.
|
||||||
|
|
||||||
|
@ -2358,3 +2339,26 @@ func ctlcmdRecalculateMailboxCounts(ctl *ctl, account string) {
|
||||||
ctl.xreadok()
|
ctl.xreadok()
|
||||||
ctl.xstreamto(os.Stdout)
|
ctl.xstreamto(os.Stdout)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func cmdMessageParse(c *cmd) {
|
||||||
|
c.params = "message.eml"
|
||||||
|
c.help = "Parse message, print JSON representation."
|
||||||
|
|
||||||
|
args := c.Parse()
|
||||||
|
if len(args) != 1 {
|
||||||
|
c.Usage()
|
||||||
|
}
|
||||||
|
|
||||||
|
f, err := os.Open(args[0])
|
||||||
|
xcheckf(err, "open")
|
||||||
|
defer f.Close()
|
||||||
|
|
||||||
|
part, err := message.Parse(f)
|
||||||
|
xcheckf(err, "parsing message")
|
||||||
|
err = part.Walk(nil)
|
||||||
|
xcheckf(err, "parsing nested parts")
|
||||||
|
enc := json.NewEncoder(os.Stdout)
|
||||||
|
enc.SetIndent("", "\t")
|
||||||
|
err = enc.Encode(part)
|
||||||
|
xcheckf(err, "write")
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue