mirror of
https://git.sr.ht/~rjarry/aerc
synced 2025-02-23 07:53:59 +01:00
data:image/s3,"s3://crabby-images/d5dcf/d5dcf1207f834c8d2925a3c46621719a00ba632b" alt="Robin Jarry"
Add a new models.UID type (an alias to string). Replace all occurrences of uint32 being used as message UID or thread UID with models.UID. Update all workers to only expose models.UID values and deal with the conversion internally. Only IMAP needs to convert these to uint32. All other backends already use plain strings as message identifiers, in which case no conversion is even needed. The directory tree implementation needed to be heavily refactored in order to accommodate thread UID not being usable as a list index. Signed-off-by: Robin Jarry <robin@jarry.cc> Tested-by: Inwit <inwit@sindominio.net> Tested-by: Tim Culverhouse <tim@timculverhouse.com>
67 lines
1.6 KiB
Go
67 lines
1.6 KiB
Go
package maildir
|
|
|
|
import (
|
|
"context"
|
|
"runtime"
|
|
"sync"
|
|
|
|
"git.sr.ht/~rjarry/aerc/lib/log"
|
|
"git.sr.ht/~rjarry/aerc/models"
|
|
"git.sr.ht/~rjarry/aerc/worker/lib"
|
|
"git.sr.ht/~rjarry/aerc/worker/types"
|
|
)
|
|
|
|
func (w *Worker) search(ctx context.Context, criteria *types.SearchCriteria) ([]models.UID, error) {
|
|
criteria.PrepareHeader()
|
|
requiredParts := lib.GetRequiredParts(criteria)
|
|
w.worker.Debugf("Required parts bitmask for search: %b", requiredParts)
|
|
|
|
keys, err := w.c.UIDs(*w.selected)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var matchedUids []models.UID
|
|
mu := sync.Mutex{}
|
|
wg := sync.WaitGroup{}
|
|
// Hard limit at 2x CPU cores
|
|
max := runtime.NumCPU() * 2
|
|
limit := make(chan struct{}, max)
|
|
for _, key := range keys {
|
|
select {
|
|
case <-ctx.Done():
|
|
return nil, context.Canceled
|
|
default:
|
|
limit <- struct{}{}
|
|
wg.Add(1)
|
|
go func(key models.UID) {
|
|
defer log.PanicHandler()
|
|
defer wg.Done()
|
|
success, err := w.searchKey(key, criteria, requiredParts)
|
|
if err != nil {
|
|
// don't return early so that we can still get some results
|
|
w.worker.Errorf("Failed to search key %d: %v", key, err)
|
|
} else if success {
|
|
mu.Lock()
|
|
matchedUids = append(matchedUids, key)
|
|
mu.Unlock()
|
|
}
|
|
<-limit
|
|
}(key)
|
|
|
|
}
|
|
}
|
|
wg.Wait()
|
|
return matchedUids, nil
|
|
}
|
|
|
|
// Execute the search criteria for the given key, returns true if search succeeded
|
|
func (w *Worker) searchKey(key models.UID, criteria *types.SearchCriteria,
|
|
parts lib.MsgParts,
|
|
) (bool, error) {
|
|
message, err := w.c.Message(*w.selected, key)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
return lib.SearchMessage(message, criteria, parts)
|
|
}
|