mirror of
https://github.com/mjl-/mox.git
synced 2025-01-28 15:25:55 +03:00
bf04fb8a1a
before, we used heuristics to decide when to train/untrain a message as junk or nonjunk: the message had to be seen, be in certain mailboxes. then if a message was marked as junk, it was junk. and otherwise it was nonjunk. this wasn't good enough: you may want to keep some messages around as neither junk or nonjunk. and that wasn't possible. ideally, we would just look at the imap $Junk and $NotJunk flags. the problem is that mail clients don't set these flags, or don't make it easy. thunderbird can set the flags based on its own bayesian filter. it has a shortcut for marking Junk and moving it to the junk folder (good), but the counterpart of notjunk only marks a message as notjunk without showing in the UI that it was marked as notjunk. there is also no "move and mark as notjunk" mechanism. e.g. "archive" does not mark a message as notjunk. ios mail and mutt don't appear to have any way to see or change the $Junk and $NotJunk flags. what email clients do have is the ability to move messages to other mailboxes/folders. so mox now has a mechanism that allows you to configure mailboxes that automatically set $Junk or $NotJunk (or clear both) when a message is moved/copied/delivered to that folder. e.g. a mailbox called junk or spam or rejects marks its messags as junk. inbox, postmaster, dmarc, tlsrpt, neutral* mark their messages as neither junk or notjunk. other folders mark their messages as notjunk. e.g. list/*, archive. this functionality is optional, but enabled with the quickstart and for new accounts. also, mox now keeps track of the previous training of a message and will only untrain/train if needed. before, there probably have been duplicate or missing (un)trainings. this also includes a new subcommand "retrain" to recreate the junkfilter for an account. you should run it after updating to this version. and you should probably also modify your account config to include the AutomaticJunkFlags.
142 lines
7.2 KiB
Go
142 lines
7.2 KiB
Go
/*
|
|
Package bstore is a database library for storing and querying Go struct data.
|
|
|
|
Bstore is designed as a small, pure Go library that still provides most of
|
|
the common data consistency requirements for modest database use cases. Bstore
|
|
aims to make basic use of cgo-based libraries, such as sqlite, unnecessary.
|
|
|
|
Bstore implements autoincrementing primary keys, indices, default values,
|
|
enforcement of nonzero, unique and referential integrity constraints, automatic
|
|
schema updates and a query API for combining filters/sorting/limits. Queries
|
|
are planned and executed using indices for fast execution where possible.
|
|
Bstores is designed with the Go type system in mind: you typically don't have to
|
|
write any (un)marshal code for your types.
|
|
|
|
# Field types
|
|
|
|
Struct field types currently supported for storing, including pointers to these
|
|
types, but not pointers to pointers:
|
|
|
|
- int (as int32), int8, int16, int32, int64
|
|
- uint (as uint32), uint8, uint16, uint32, uint64
|
|
- bool, float32, float64, string, []byte
|
|
- Maps, with keys and values of any supported type, except keys with pointer types.
|
|
- Slices, with elements of any supported type.
|
|
- time.Time
|
|
- Types that implement binary.MarshalBinary and binary.UnmarshalBinary, useful
|
|
for struct types with state in private fields. Do not change the
|
|
(Un)marshalBinary method in an incompatible way without a data migration.
|
|
- Structs, with fields of any supported type.
|
|
|
|
Note: int and uint are stored as int32 and uint32, for compatibility of database
|
|
files between 32bit and 64bit systems. Where possible, use explicit (u)int32 or
|
|
(u)int64 types.
|
|
|
|
Embedded structs are handled by storing the individual fields of the embedded
|
|
struct. The named embedded type is not part of the type schema, and can
|
|
currently only be used with UpdateField and UpdateFields, not for filtering.
|
|
|
|
Bstore embraces the use of Go zero values. Use zero values, possibly pointers,
|
|
where you would use NULL values in SQL.
|
|
|
|
Types that have not yet been implemented: interface values, (fixed length) arrays,
|
|
complex numbers.
|
|
|
|
# Struct tags
|
|
|
|
The typical Go struct can be stored in the database. The first field of a
|
|
struct type is its primary key, and must always be unique. Additional behaviour
|
|
can be configured through struct tag "bstore". The values are comma-separated.
|
|
Typically one word, but some have multiple space-separated words:
|
|
|
|
- "-" ignores the field entirely.
|
|
- "name <fieldname>", use "fieldname" instead of the Go type field name.
|
|
- "nonzero", enforces that field values are not the zero value.
|
|
- "noauto", only valid for integer types, and only for the primary key. By
|
|
default, an integer-typed primary key will automatically get a next value
|
|
assigned on insert when it is 0. With noauto inserting a 0 value results in an
|
|
error. For primary keys of other types inserting the zero value always results
|
|
in an error.
|
|
- "index" or "index <field1+field2+...> [<name>]", adds an index. In the first
|
|
form, the index is on the field on which the tag is specified, and the index
|
|
name is the same as the field name. In the second form multiple fields can be
|
|
specified, and an optional name. The first field must be the field on which
|
|
the tag is specified. The field names are +-separated. The default name for
|
|
the second form is the same +-separated string but can be set explicitly to
|
|
the second parameter. An index can only be set for basic integer types, bools,
|
|
time and strings. Indices are automatically (re)created when registering a
|
|
type.
|
|
- "unique" or "unique <field1+field2+...> [<name>]", adds an index as with
|
|
"index" and also enforces a unique constraint. For time.Time the timezone is
|
|
ignored for the uniqueness check.
|
|
- "ref <type>", enforces that the value exists as primary key for "type".
|
|
Field types must match exactly, e.g. you cannot reference an int with an int64.
|
|
An index is automatically created and maintained for fields with a foreign key,
|
|
for efficiently checking that removed records in the referenced type are not in
|
|
use. If the field has the zero value, the reference is not checked. If you
|
|
require a valid reference, add "nonzero".
|
|
- "default <value>", replaces a zero value with the specified value on record
|
|
insert. Special value "now" is recognized for time.Time as the current time.
|
|
Times are parsed as time.RFC3339 otherwise. Supported types: bool
|
|
("true"/"false"), integers, floats, strings. Value is not quoted and no escaping
|
|
of special characters, like the comma that separates struct tag words, is
|
|
possible. Defaults are also replaced on fields in nested structs and
|
|
slices, but not in maps.
|
|
- "typename <name>", override name of the type. The name of the Go type is
|
|
used by default. Can only be present on the first field (primary key).
|
|
Useful for doing schema updates.
|
|
|
|
# Schema updates
|
|
|
|
Before using a Go type, you must register it for use with the open database by
|
|
passing a (zero) value of that type to the Open or Register functions. For each
|
|
type, a type definition is stored in the database. If a type has an updated
|
|
definition since the previous database open, a new type definition is added to
|
|
the database automatically and any required modifications are made: Indexes
|
|
(re)created, fields added/removed, new nonzero/unique/reference constraints
|
|
validated.
|
|
|
|
If data/types cannot be updated automatically (e.g. converting an int field into
|
|
a string field), custom data migration code is needed. You may have to keep
|
|
track of a data/schema version.
|
|
|
|
As a special case, you can switch field types between pointer and non-pointer
|
|
types. With one exception: changing from pointer to non-pointer where the type
|
|
has a field that must be nonzero is not allowed. The on-disk encoding will not be
|
|
changed, and nil pointers will turn into zero values, and zero values into nil
|
|
pointers. Also see section Limitations about pointer types.
|
|
|
|
Because named embed structs are not part of the type definition, you can
|
|
wrap/unwrap fields into a embed/anonymous struct field. No new type definition
|
|
is created.
|
|
|
|
# BoltDB
|
|
|
|
BoltDB is used as underlying storage. Bolt provides ACID transactions, storing
|
|
its data in a B+tree. Only a single write transaction can be active at a time,
|
|
but otherwise multiple read-only transactions can be active. Do not start a
|
|
blocking read-only transaction while holding a writable transaction or vice
|
|
versa, this will cause deadlock.
|
|
|
|
Bolt uses Go types that are memory mapped to the database file. This means bolt
|
|
database files cannot be transferred between machines with different endianness.
|
|
Bolt uses explicit widths for its types, so files can be transferred between
|
|
32bit and 64bit machines of same endianness.
|
|
|
|
# Limitations
|
|
|
|
Bstore does not implement the equivalent of SQL joins, aggregates, and many
|
|
other concepts.
|
|
|
|
Filtering/comparing/sorting on pointer fields is not currently allowed. Pointer
|
|
fields cannot have a (unique) index due to the current index format. Using zero
|
|
values is recommended instead for now.
|
|
|
|
Integer field types can be expanded to wider types, but not to a different
|
|
signedness or a smaller integer (fewer bits). The primary key of a type cannot
|
|
currently be changed.
|
|
|
|
The first field of a struct is always the primary key. Types requires an
|
|
explicit primary key. Autoincrement is only available for the primary key.
|
|
*/
|
|
package bstore
|