fix(sessionresolver): proxy check conditional on existing proxy (#264)

There was a face-palming error in the implementation causing the proxy
check to be implemented also without a proxy.

This meant that we were ALWAYS skipping http3 and system resolvers.

The bug has been introduced in 3.8.0. So, the currently released
version of the probe, sadly, has this beheavior :-(.

Reference issue https://github.com/ooni/probe/issues/1426.
This commit is contained in:
Simone Basso 2021-03-25 15:18:29 +01:00 committed by GitHub
parent 3b029ee0d6
commit c94721d9e5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 60 additions and 13 deletions

View File

@ -97,7 +97,7 @@ func (r *Resolver) newresolver(URL string) (childResolver, error) {
func (r *Resolver) getresolver(URL string) (childResolver, error) { func (r *Resolver) getresolver(URL string) (childResolver, error) {
defer r.mu.Unlock() defer r.mu.Unlock()
r.mu.Lock() r.mu.Lock()
if re, found := r.res[URL]; found == true { if re, found := r.res[URL]; found {
return re, nil // already created return re, nil // already created
} }
re, err := r.newresolver(URL) re, err := r.newresolver(URL)

View File

@ -17,6 +17,10 @@
// but it will of course be the most popular resolver if anything else // but it will of course be the most popular resolver if anything else
// is failing us. (We will still occasionally probe for other working // is failing us. (We will still occasionally probe for other working
// resolvers and increase their score on success.) // resolvers and increase their score on success.)
//
// We also support a socks5 proxy. When such a proxy is configured,
// the code WILL skip http3 resolvers AS WELL AS the system
// resolver, in an attempt to avoid leaking your queries.
package sessionresolver package sessionresolver
import ( import (
@ -34,17 +38,59 @@ import (
"github.com/ooni/probe-cli/v3/internal/engine/runtimex" "github.com/ooni/probe-cli/v3/internal/engine/runtimex"
) )
// Resolver is the session resolver. You should create an instance of // Resolver is the session resolver. Resolver will try to use
// this structure and use it in session.go. // a bunch of DoT/DoH resolvers before falling back to the
// system resolver. The relative priorities of the resolver
// are stored onto the KVStore such that we can remember them
// and therefore we can generally give preference to underlying
// DoT/DoH resolvers that work better.
//
// You MUST NOT modify public fields of this structure once it
// has been created, because that MAY lead to data races.
//
// You should create an instance of this structure and use
// it in internal/engine/session.go.
type Resolver struct { type Resolver struct {
ByteCounter *bytecounter.Counter // optional // ByteCounter is the optional byte counter. It will count
KVStore KVStore // optional // the bytes used by any child resolver except for the
Logger Logger // optional // system resolver, whose bytes ARE NOT counted. If this
ProxyURL *url.URL // optional // field is not set, then we won't count the bytes.
ByteCounter *bytecounter.Counter
// KVStore is the optional key-value store where you
// want us to write statistics about which resolver is
// working better in your network. If this field is
// not set, then we'll use a in-memory store.
KVStore KVStore
// Logger is the optional logger you want us to use
// to emit log messages.
Logger Logger
// ProxyURL is the optional URL of the socks5 proxy
// we should be using. If not set, then we WON'T use
// any proxy. If set, then we WON'T use any http3
// based resolvers and we WON'T use the system resolver.
ProxyURL *url.URL
// codec is the optional codec to use. If not set, we
// will construct a default codec.
codec codec codec codec
// dnsClientMaker is the optional dnsclientmaker to
// use. If not set, we will use the default.
dnsClientMaker dnsclientmaker dnsClientMaker dnsclientmaker
// mu provides synchronisation of internal fields.
mu sync.Mutex mu sync.Mutex
// once ensures that CloseIdleConnection is
// run just once.
once sync.Once once sync.Once
// res maps a URL to a child resolver. We will
// construct child resolvers just once and we
// will track them into this field.
res map[string]childResolver res map[string]childResolver
} }
@ -73,7 +119,8 @@ func (r *Resolver) LookupHost(ctx context.Context, hostname string) ([]string, e
defer r.writestate(state) defer r.writestate(state)
me := multierror.New(ErrLookupHost) me := multierror.New(ErrLookupHost)
for _, e := range state { for _, e := range state {
if r.shouldSkipWithProxy(e) { if r.ProxyURL != nil && r.shouldSkipWithProxy(e) {
r.logger().Infof("sessionresolver: skipping with proxy: %+v", e)
continue // we cannot proxy this URL so ignore it continue // we cannot proxy this URL so ignore it
} }
addrs, err := r.lookupHost(ctx, e, hostname) addrs, err := r.lookupHost(ctx, e, hostname)

View File

@ -294,7 +294,7 @@ func TestResolverWorksWithProxy(t *testing.T) {
<-done <-done
// check results // check results
if !errors.Is(err, ErrLookupHost) { if !errors.Is(err, ErrLookupHost) {
t.Fatal("not the error we expected") t.Fatal("not the error we expected", err)
} }
if addrs != nil { if addrs != nil {
t.Fatal("expected nil addrs") t.Fatal("expected nil addrs")