package store import ( "errors" "os" "path/filepath" "github.com/mjl-/bstore" "github.com/mjl-/mox/config" "github.com/mjl-/mox/junk" "github.com/mjl-/mox/mlog" "github.com/mjl-/mox/mox-" ) // ErrNoJunkFilter indicates user did not configure/enable a junk filter. var ErrNoJunkFilter = errors.New("junkfilter: not configured") // OpenJunkFilter returns an opened junk filter for the account. // If the account does not have a junk filter enabled, ErrNotConfigured is returned. // Do not forget to save the filter after modifying, and to always close the filter when done. // An empty filter is initialized on first access of the filter. func (a *Account) OpenJunkFilter(log *mlog.Log) (*junk.Filter, *config.JunkFilter, error) { conf, ok := mox.Conf.Account(a.Name) if !ok { return nil, nil, ErrAccountUnknown } jf := conf.JunkFilter if jf == nil { return nil, jf, ErrNoJunkFilter } basePath := mox.DataDirPath("accounts") dbPath := filepath.Join(basePath, a.Name, "junkfilter.db") bloomPath := filepath.Join(basePath, a.Name, "junkfilter.bloom") if _, xerr := os.Stat(dbPath); xerr != nil && os.IsNotExist(xerr) { f, err := junk.NewFilter(log, jf.Params, dbPath, bloomPath) return f, jf, err } f, err := junk.OpenFilter(log, jf.Params, dbPath, bloomPath, false) return f, jf, err } // RetrainMessages (un)trains messages, if relevant given their flags. Updates // m.TrainedJunk after retraining. func (a *Account) RetrainMessages(log *mlog.Log, tx *bstore.Tx, msgs []Message, absentOK bool) (rerr error) { if len(msgs) == 0 { return nil } var jf *junk.Filter for i := range msgs { if !msgs[i].NeedsTraining() { continue } // Lazy open the junk filter. if jf == nil { var err error jf, _, err = a.OpenJunkFilter(log) if err != nil && errors.Is(err, ErrNoJunkFilter) { // No junk filter configured. Nothing more to do. return nil } defer func() { if jf != nil { err := jf.Close() if rerr == nil { rerr = err } } }() } if err := a.RetrainMessage(log, tx, jf, &msgs[i], absentOK); err != nil { return err } } return nil } // RetrainMessage untrains and/or trains a message, if relevant given m.TrainedJunk // and m.Junk/m.Notjunk. Updates m.TrainedJunk after retraining. func (a *Account) RetrainMessage(log *mlog.Log, tx *bstore.Tx, jf *junk.Filter, m *Message, absentOK bool) error { untrain := m.TrainedJunk != nil untrainJunk := untrain && *m.TrainedJunk train := m.Junk || m.Notjunk && !(m.Junk && m.Notjunk) trainJunk := m.Junk if !untrain && !train || (untrain && train && untrainJunk == trainJunk) { return nil } log.Info("updating junk filter", mlog.Field("untrain", untrain), mlog.Field("untrainJunk", untrainJunk), mlog.Field("train", train), mlog.Field("trainJunk", trainJunk)) mr := a.MessageReader(*m) defer mr.Close() p, err := m.LoadPart(mr) if err != nil { log.Errorx("loading part for message", err) return nil } words, err := jf.ParseMessage(p) if err != nil { log.Errorx("parsing message for updating junk filter", err, mlog.Field("parse", "")) return nil } if untrain { err := jf.Untrain(!untrainJunk, words) if err != nil { return err } m.TrainedJunk = nil } if train { err := jf.Train(!trainJunk, words) if err != nil { return err } m.TrainedJunk = &trainJunk } if err := tx.Update(m); err != nil && (!absentOK || err != bstore.ErrAbsent) { return err } return nil } // TrainMessage trains the junk filter based on the current m.Junk/m.Notjunk flags, // disregarding m.TrainedJunk and not updating that field. func (a *Account) TrainMessage(log *mlog.Log, jf *junk.Filter, m Message) (bool, error) { if !m.Junk && !m.Notjunk || (m.Junk && m.Notjunk) { return false, nil } mr := a.MessageReader(m) defer mr.Close() p, err := m.LoadPart(mr) if err != nil { log.Errorx("loading part for message", err) return false, nil } words, err := jf.ParseMessage(p) if err != nil { log.Errorx("parsing message for updating junk filter", err, mlog.Field("parse", "")) return false, nil } return true, jf.Train(m.Notjunk, words) }