ooni-probe-cli/internal/engine/internal/sessionresolver/state.go
Simone Basso 034db78f94
refactor(sessionresolver): adapt to changing network conditions (#238)
* feat(sessionresolver): try many and use what works

* fix(sessionresolver): make sure we can use quic

* fix: the config struct is unnecessary

* fix: make kvstore optional

* feat: write simple integration test

* feat: start adding tests

* feat: continue writing tests

* fix(sessionresolver): add more unit tests

* fix(sessionresolver): finish adding tests

* refactor(sessionresolver): changes after code review
2021-03-03 11:28:39 +01:00

94 lines
2.2 KiB
Go

package sessionresolver
import (
"errors"
"sort"
)
// storekey is the key used by the key value store to store
// the state required by this package.
const storekey = "sessionresolver.state"
// resolverinfo contains info about a resolver.
type resolverinfo struct {
// URL is the URL of a resolver.
URL string
// Score is the score of a resolver.
Score float64
}
// readstate reads the resolver state from disk
func (r *Resolver) readstate() ([]*resolverinfo, error) {
data, err := r.kvstore().Get(storekey)
if err != nil {
return nil, err
}
var ri []*resolverinfo
if err := r.getCodec().Decode(data, &ri); err != nil {
return nil, err
}
return ri, nil
}
// errNoEntries indicates that no entry remained after we pruned
// all the available entries in readstateandprune.
var errNoEntries = errors.New("sessionresolver: no available entries")
// readstateandprune reads the state from disk and removes all the
// entries that we don't actually support.
func (r *Resolver) readstateandprune() ([]*resolverinfo, error) {
ri, err := r.readstate()
if err != nil {
return nil, err
}
var out []*resolverinfo
for _, e := range ri {
if _, found := allbyurl[e.URL]; !found {
continue // we don't support this specific entry
}
out = append(out, e)
}
if len(out) <= 0 {
return nil, errNoEntries
}
return out, nil
}
// sortstate sorts the state by descending score
func sortstate(ri []*resolverinfo) {
sort.SliceStable(ri, func(i, j int) bool {
return ri[i].Score >= ri[j].Score
})
}
// readstatedefault reads the state from disk and merges the state
// so that all supported entries are represented.
func (r *Resolver) readstatedefault() []*resolverinfo {
ri, _ := r.readstateandprune()
here := make(map[string]bool)
for _, e := range ri {
here[e.URL] = true // record what we already have
}
for _, e := range allmakers {
if _, found := here[e.url]; found {
continue // already here so no need to add
}
ri = append(ri, &resolverinfo{
URL: e.url,
Score: e.score,
})
}
sortstate(ri)
return ri
}
// writestate writes the state on the kvstore.
func (r *Resolver) writestate(ri []*resolverinfo) error {
data, err := r.getCodec().Encode(ri)
if err != nil {
return err
}
return r.kvstore().Set(storekey, data)
}