2021-02-02 12:05:47 +01:00
|
|
|
// Package dnscheck contains the DNS check experiment.
|
|
|
|
//
|
|
|
|
// See https://github.com/ooni/spec/blob/master/nettests/ts-028-dnscheck.md.
|
|
|
|
package dnscheck
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"net"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
refactor: flatten and separate (#353)
* refactor(atomicx): move outside the engine package
After merging probe-engine into probe-cli, my impression is that we have
too much unnecessary nesting of packages in this repository.
The idea of this commit and of a bunch of following commits will instead
be to reduce the nesting and simplify the structure.
While there, improve the documentation.
* fix: always use the atomicx package
For consistency, never use sync/atomic and always use ./internal/atomicx
so we can just grep and make sure we're not risking to crash if we make
a subtle mistake on a 32 bit platform.
While there, mention in the contributing guidelines that we want to
always prefer the ./internal/atomicx package over sync/atomic.
* fix(atomicx): remove unnecessary constructor
We don't need a constructor here. The default constructed `&Int64{}`
instance is already usable and the constructor does not add anything to
what we are doing, rather it just creates extra confusion.
* cleanup(atomicx): we are not using Float64
Because atomicx.Float64 is unused, we can safely zap it.
* cleanup(atomicx): simplify impl and improve tests
We can simplify the implementation by using defer and by letting
the Load() method call Add(0).
We can improve tests by making many goroutines updated the
atomic int64 value concurrently.
* refactor(fsx): can live in the ./internal pkg
Let us reduce the amount of nesting. While there, ensure that the
package only exports the bare minimum, and improve the documentation
of the tests, to ease reading the code.
* refactor: move runtimex to ./internal
* refactor: move shellx into the ./internal package
While there, remove unnecessary dependency between packages.
While there, specify in the contributing guidelines that
one should use x/sys/execabs instead of os/exec.
* refactor: move ooapi into the ./internal pkg
* refactor(humanize): move to ./internal and better docs
* refactor: move platform to ./internal
* refactor(randx): move to ./internal
* refactor(multierror): move into the ./internal pkg
* refactor(kvstore): all kvstores in ./internal
Rather than having part of the kvstore inside ./internal/engine/kvstore
and part in ./internal/engine/kvstore.go, let us put every piece of code
that is kvstore related into the ./internal/kvstore package.
* fix(kvstore): always return ErrNoSuchKey on Get() error
It should help to use the kvstore everywhere removing all the
copies that are lingering around the tree.
* sessionresolver: make KVStore mandatory
Simplifies implementation. While there, use the ./internal/kvstore
package rather than having our private implementation.
* fix(ooapi): use the ./internal/kvstore package
* fix(platform): better documentation
2021-06-04 10:34:18 +02:00
|
|
|
"github.com/ooni/probe-cli/v3/internal/atomicx"
|
2021-02-02 12:05:47 +01:00
|
|
|
"github.com/ooni/probe-cli/v3/internal/engine/experiment/urlgetter"
|
|
|
|
"github.com/ooni/probe-cli/v3/internal/engine/netx"
|
2022-01-03 13:53:23 +01:00
|
|
|
"github.com/ooni/probe-cli/v3/internal/model"
|
refactor: flatten and separate (#353)
* refactor(atomicx): move outside the engine package
After merging probe-engine into probe-cli, my impression is that we have
too much unnecessary nesting of packages in this repository.
The idea of this commit and of a bunch of following commits will instead
be to reduce the nesting and simplify the structure.
While there, improve the documentation.
* fix: always use the atomicx package
For consistency, never use sync/atomic and always use ./internal/atomicx
so we can just grep and make sure we're not risking to crash if we make
a subtle mistake on a 32 bit platform.
While there, mention in the contributing guidelines that we want to
always prefer the ./internal/atomicx package over sync/atomic.
* fix(atomicx): remove unnecessary constructor
We don't need a constructor here. The default constructed `&Int64{}`
instance is already usable and the constructor does not add anything to
what we are doing, rather it just creates extra confusion.
* cleanup(atomicx): we are not using Float64
Because atomicx.Float64 is unused, we can safely zap it.
* cleanup(atomicx): simplify impl and improve tests
We can simplify the implementation by using defer and by letting
the Load() method call Add(0).
We can improve tests by making many goroutines updated the
atomic int64 value concurrently.
* refactor(fsx): can live in the ./internal pkg
Let us reduce the amount of nesting. While there, ensure that the
package only exports the bare minimum, and improve the documentation
of the tests, to ease reading the code.
* refactor: move runtimex to ./internal
* refactor: move shellx into the ./internal package
While there, remove unnecessary dependency between packages.
While there, specify in the contributing guidelines that
one should use x/sys/execabs instead of os/exec.
* refactor: move ooapi into the ./internal pkg
* refactor(humanize): move to ./internal and better docs
* refactor: move platform to ./internal
* refactor(randx): move to ./internal
* refactor(multierror): move into the ./internal pkg
* refactor(kvstore): all kvstores in ./internal
Rather than having part of the kvstore inside ./internal/engine/kvstore
and part in ./internal/engine/kvstore.go, let us put every piece of code
that is kvstore related into the ./internal/kvstore package.
* fix(kvstore): always return ErrNoSuchKey on Get() error
It should help to use the kvstore everywhere removing all the
copies that are lingering around the tree.
* sessionresolver: make KVStore mandatory
Simplifies implementation. While there, use the ./internal/kvstore
package rather than having our private implementation.
* fix(ooapi): use the ./internal/kvstore package
* fix(platform): better documentation
2021-06-04 10:34:18 +02:00
|
|
|
"github.com/ooni/probe-cli/v3/internal/runtimex"
|
2022-06-02 00:50:55 +02:00
|
|
|
"github.com/ooni/probe-cli/v3/internal/tracex"
|
2021-02-02 12:05:47 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
testName = "dnscheck"
|
2022-09-02 13:31:24 +02:00
|
|
|
testVersion = "0.9.2"
|
2021-02-02 12:05:47 +01:00
|
|
|
defaultDomain = "example.org"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Endpoints keeps track of repeatedly measured endpoints.
|
|
|
|
type Endpoints struct {
|
|
|
|
WaitTime time.Duration
|
refactor: flatten and separate (#353)
* refactor(atomicx): move outside the engine package
After merging probe-engine into probe-cli, my impression is that we have
too much unnecessary nesting of packages in this repository.
The idea of this commit and of a bunch of following commits will instead
be to reduce the nesting and simplify the structure.
While there, improve the documentation.
* fix: always use the atomicx package
For consistency, never use sync/atomic and always use ./internal/atomicx
so we can just grep and make sure we're not risking to crash if we make
a subtle mistake on a 32 bit platform.
While there, mention in the contributing guidelines that we want to
always prefer the ./internal/atomicx package over sync/atomic.
* fix(atomicx): remove unnecessary constructor
We don't need a constructor here. The default constructed `&Int64{}`
instance is already usable and the constructor does not add anything to
what we are doing, rather it just creates extra confusion.
* cleanup(atomicx): we are not using Float64
Because atomicx.Float64 is unused, we can safely zap it.
* cleanup(atomicx): simplify impl and improve tests
We can simplify the implementation by using defer and by letting
the Load() method call Add(0).
We can improve tests by making many goroutines updated the
atomic int64 value concurrently.
* refactor(fsx): can live in the ./internal pkg
Let us reduce the amount of nesting. While there, ensure that the
package only exports the bare minimum, and improve the documentation
of the tests, to ease reading the code.
* refactor: move runtimex to ./internal
* refactor: move shellx into the ./internal package
While there, remove unnecessary dependency between packages.
While there, specify in the contributing guidelines that
one should use x/sys/execabs instead of os/exec.
* refactor: move ooapi into the ./internal pkg
* refactor(humanize): move to ./internal and better docs
* refactor: move platform to ./internal
* refactor(randx): move to ./internal
* refactor(multierror): move into the ./internal pkg
* refactor(kvstore): all kvstores in ./internal
Rather than having part of the kvstore inside ./internal/engine/kvstore
and part in ./internal/engine/kvstore.go, let us put every piece of code
that is kvstore related into the ./internal/kvstore package.
* fix(kvstore): always return ErrNoSuchKey on Get() error
It should help to use the kvstore everywhere removing all the
copies that are lingering around the tree.
* sessionresolver: make KVStore mandatory
Simplifies implementation. While there, use the ./internal/kvstore
package rather than having our private implementation.
* fix(ooapi): use the ./internal/kvstore package
* fix(platform): better documentation
2021-06-04 10:34:18 +02:00
|
|
|
count *atomicx.Int64
|
2021-02-02 12:05:47 +01:00
|
|
|
nextVisit map[string]time.Time
|
|
|
|
mu sync.Mutex
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *Endpoints) maybeSleep(resolverURL string, logger model.Logger) {
|
|
|
|
if e == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
defer e.mu.Unlock()
|
|
|
|
e.mu.Lock()
|
|
|
|
nextTime, found := e.nextVisit[resolverURL]
|
|
|
|
now := time.Now()
|
|
|
|
if !found || now.After(nextTime) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
sleepTime := nextTime.Sub(now)
|
refactor: flatten and separate (#353)
* refactor(atomicx): move outside the engine package
After merging probe-engine into probe-cli, my impression is that we have
too much unnecessary nesting of packages in this repository.
The idea of this commit and of a bunch of following commits will instead
be to reduce the nesting and simplify the structure.
While there, improve the documentation.
* fix: always use the atomicx package
For consistency, never use sync/atomic and always use ./internal/atomicx
so we can just grep and make sure we're not risking to crash if we make
a subtle mistake on a 32 bit platform.
While there, mention in the contributing guidelines that we want to
always prefer the ./internal/atomicx package over sync/atomic.
* fix(atomicx): remove unnecessary constructor
We don't need a constructor here. The default constructed `&Int64{}`
instance is already usable and the constructor does not add anything to
what we are doing, rather it just creates extra confusion.
* cleanup(atomicx): we are not using Float64
Because atomicx.Float64 is unused, we can safely zap it.
* cleanup(atomicx): simplify impl and improve tests
We can simplify the implementation by using defer and by letting
the Load() method call Add(0).
We can improve tests by making many goroutines updated the
atomic int64 value concurrently.
* refactor(fsx): can live in the ./internal pkg
Let us reduce the amount of nesting. While there, ensure that the
package only exports the bare minimum, and improve the documentation
of the tests, to ease reading the code.
* refactor: move runtimex to ./internal
* refactor: move shellx into the ./internal package
While there, remove unnecessary dependency between packages.
While there, specify in the contributing guidelines that
one should use x/sys/execabs instead of os/exec.
* refactor: move ooapi into the ./internal pkg
* refactor(humanize): move to ./internal and better docs
* refactor: move platform to ./internal
* refactor(randx): move to ./internal
* refactor(multierror): move into the ./internal pkg
* refactor(kvstore): all kvstores in ./internal
Rather than having part of the kvstore inside ./internal/engine/kvstore
and part in ./internal/engine/kvstore.go, let us put every piece of code
that is kvstore related into the ./internal/kvstore package.
* fix(kvstore): always return ErrNoSuchKey on Get() error
It should help to use the kvstore everywhere removing all the
copies that are lingering around the tree.
* sessionresolver: make KVStore mandatory
Simplifies implementation. While there, use the ./internal/kvstore
package rather than having our private implementation.
* fix(ooapi): use the ./internal/kvstore package
* fix(platform): better documentation
2021-06-04 10:34:18 +02:00
|
|
|
if e.count == nil {
|
|
|
|
e.count = &atomicx.Int64{}
|
|
|
|
}
|
|
|
|
e.count.Add(1)
|
2021-02-02 12:05:47 +01:00
|
|
|
logger.Infof("waiting %v before testing %s again", sleepTime, resolverURL)
|
|
|
|
time.Sleep(sleepTime)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *Endpoints) maybeRegister(resolverURL string) {
|
|
|
|
if e != nil && !strings.HasPrefix(resolverURL, "udp://") {
|
|
|
|
defer e.mu.Unlock()
|
|
|
|
e.mu.Lock()
|
|
|
|
if e.nextVisit == nil {
|
|
|
|
e.nextVisit = make(map[string]time.Time)
|
|
|
|
}
|
|
|
|
waitTime := 180 * time.Second
|
|
|
|
if e.WaitTime > 0 {
|
|
|
|
waitTime = e.WaitTime
|
|
|
|
}
|
|
|
|
e.nextVisit[resolverURL] = time.Now().Add(waitTime)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Config contains the experiment's configuration.
|
|
|
|
type Config struct {
|
|
|
|
DefaultAddrs string `json:"default_addrs" ooni:"default addresses for domain"`
|
|
|
|
Domain string `json:"domain" ooni:"domain to resolve using the specified resolver"`
|
|
|
|
HTTP3Enabled bool `json:"http3_enabled" ooni:"use http3 instead of http/1.1 or http2"`
|
|
|
|
HTTPHost string `json:"http_host" ooni:"force using specific HTTP Host header"`
|
|
|
|
TLSServerName string `json:"tls_server_name" ooni:"force TLS to using a specific SNI in Client Hello"`
|
|
|
|
TLSVersion string `json:"tls_version" ooni:"Force specific TLS version (e.g. 'TLSv1.3')"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestKeys contains the results of the dnscheck experiment.
|
|
|
|
type TestKeys struct {
|
|
|
|
DefaultAddrs string `json:"x_default_addrs"`
|
|
|
|
Domain string `json:"domain"`
|
|
|
|
HTTP3Enabled bool `json:"x_http3_enabled,omitempty"`
|
|
|
|
HTTPHost string `json:"x_http_host,omitempty"`
|
|
|
|
TLSServerName string `json:"x_tls_server_name,omitempty"`
|
|
|
|
TLSVersion string `json:"x_tls_version,omitempty"`
|
2022-09-02 13:31:24 +02:00
|
|
|
Residual bool `json:"x_residual"`
|
2021-02-02 12:05:47 +01:00
|
|
|
Bootstrap *urlgetter.TestKeys `json:"bootstrap"`
|
|
|
|
BootstrapFailure *string `json:"bootstrap_failure"`
|
|
|
|
Lookups map[string]urlgetter.TestKeys `json:"lookups"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// Measurer performs the measurement.
|
|
|
|
type Measurer struct {
|
|
|
|
Config
|
|
|
|
Endpoints *Endpoints
|
|
|
|
}
|
|
|
|
|
|
|
|
// ExperimentName implements model.ExperimentSession.ExperimentName
|
|
|
|
func (m *Measurer) ExperimentName() string {
|
|
|
|
return testName
|
|
|
|
}
|
|
|
|
|
|
|
|
// ExperimentVersion implements model.ExperimentSession.ExperimentVersion
|
|
|
|
func (m *Measurer) ExperimentVersion() string {
|
|
|
|
return testVersion
|
|
|
|
}
|
|
|
|
|
|
|
|
// The following errors may be returned by this experiment. Of course these
|
|
|
|
// errors are in addition to any other errors returned by the low level packages
|
|
|
|
// that are used by this experiment to implement its functionality.
|
|
|
|
var (
|
|
|
|
ErrInputRequired = errors.New("this experiment needs input")
|
|
|
|
ErrInvalidURL = errors.New("the input URL is invalid")
|
|
|
|
ErrUnsupportedURLScheme = errors.New("unsupported URL scheme")
|
|
|
|
)
|
|
|
|
|
|
|
|
// Run implements model.ExperimentSession.Run
|
2022-11-22 10:43:47 +01:00
|
|
|
func (m *Measurer) Run(ctx context.Context, args *model.ExperimentArgs) error {
|
|
|
|
_ = args.Callbacks
|
|
|
|
measurement := args.Measurement
|
|
|
|
sess := args.Session
|
|
|
|
|
2021-02-02 12:05:47 +01:00
|
|
|
// 1. fill the measurement with test keys
|
|
|
|
tk := new(TestKeys)
|
|
|
|
tk.Lookups = make(map[string]urlgetter.TestKeys)
|
|
|
|
measurement.TestKeys = tk
|
|
|
|
urlgetter.RegisterExtensions(measurement)
|
|
|
|
|
|
|
|
// 2. select the domain to resolve or use default and, while there, also
|
|
|
|
// ensure that we register all the other options we're using.
|
|
|
|
domain := m.Config.Domain
|
|
|
|
if domain == "" {
|
|
|
|
domain = defaultDomain
|
|
|
|
}
|
|
|
|
tk.DefaultAddrs = m.Config.DefaultAddrs
|
|
|
|
tk.Domain = domain
|
|
|
|
tk.HTTP3Enabled = m.Config.HTTP3Enabled
|
|
|
|
tk.HTTPHost = m.Config.HTTPHost
|
|
|
|
tk.TLSServerName = m.Config.TLSServerName
|
|
|
|
tk.TLSVersion = m.Config.TLSVersion
|
2022-09-02 13:31:24 +02:00
|
|
|
tk.Residual = m.Endpoints != nil
|
2021-02-02 12:05:47 +01:00
|
|
|
|
|
|
|
// 3. parse the input URL describing the resolver to use
|
|
|
|
input := string(measurement.Input)
|
|
|
|
if input == "" {
|
|
|
|
return ErrInputRequired
|
|
|
|
}
|
|
|
|
URL, err := url.Parse(input)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("%w: %s", ErrInvalidURL, err.Error())
|
|
|
|
}
|
|
|
|
switch URL.Scheme {
|
|
|
|
case "https", "dot", "udp", "tcp":
|
|
|
|
// all good
|
|
|
|
default:
|
|
|
|
return ErrUnsupportedURLScheme
|
|
|
|
}
|
|
|
|
|
2022-01-07 13:17:20 +01:00
|
|
|
// Implementation note: we must not return an error from now now. Returning an
|
|
|
|
// error means that we don't have a measurement to submit.
|
|
|
|
|
2021-02-02 12:05:47 +01:00
|
|
|
// 4. possibly expand a domain to a list of IP addresses.
|
|
|
|
//
|
|
|
|
// Implementation note: because the resolver we constructed also deals
|
|
|
|
// with IP addresses successfully, we just get back the IPs when we are
|
|
|
|
// passing as input an IP address rather than a domain name.
|
|
|
|
begin := measurement.MeasurementStartTimeSaved
|
2022-05-31 21:53:01 +02:00
|
|
|
evsaver := new(tracex.Saver)
|
2021-02-02 12:05:47 +01:00
|
|
|
resolver := netx.NewResolver(netx.Config{
|
|
|
|
BogonIsError: true,
|
|
|
|
Logger: sess.Logger(),
|
2022-06-02 18:18:49 +02:00
|
|
|
Saver: evsaver,
|
2021-02-02 12:05:47 +01:00
|
|
|
})
|
|
|
|
addrs, err := m.lookupHost(ctx, URL.Hostname(), resolver)
|
2022-05-31 21:53:01 +02:00
|
|
|
queries := tracex.NewDNSQueriesList(begin, evsaver.Read())
|
|
|
|
tk.BootstrapFailure = tracex.NewFailure(err)
|
2021-02-02 12:05:47 +01:00
|
|
|
if len(queries) > 0 {
|
|
|
|
// We get no queries in case we are resolving an IP address, since
|
|
|
|
// the address resolver doesn't generate events
|
|
|
|
tk.Bootstrap = &urlgetter.TestKeys{Queries: queries}
|
|
|
|
}
|
|
|
|
|
|
|
|
// 5. merge default addresses for the domain with the ones that
|
|
|
|
// we did discover here and measure them all.
|
|
|
|
allAddrs := make(map[string]bool)
|
|
|
|
for _, addr := range addrs {
|
|
|
|
allAddrs[addr] = true
|
|
|
|
}
|
|
|
|
for _, addr := range strings.Split(m.Config.DefaultAddrs, " ") {
|
|
|
|
if addr != "" {
|
|
|
|
allAddrs[addr] = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// 6. determine all the domain lookups we need to perform
|
|
|
|
const maxParallelism = 10
|
|
|
|
parallelism := maxParallelism
|
|
|
|
if parallelism > len(allAddrs) {
|
|
|
|
parallelism = len(allAddrs)
|
|
|
|
}
|
|
|
|
var inputs []urlgetter.MultiInput
|
|
|
|
multi := urlgetter.Multi{Begin: begin, Parallelism: parallelism, Session: sess}
|
|
|
|
for addr := range allAddrs {
|
|
|
|
inputs = append(inputs, urlgetter.MultiInput{
|
|
|
|
Config: urlgetter.Config{
|
|
|
|
DNSHTTPHost: m.httpHost(URL.Host),
|
|
|
|
DNSTLSServerName: m.tlsServerName(URL.Hostname()),
|
|
|
|
DNSTLSVersion: m.Config.TLSVersion,
|
|
|
|
HTTP3Enabled: m.Config.HTTP3Enabled,
|
|
|
|
RejectDNSBogons: true, // bogons are errors in this context
|
|
|
|
ResolverURL: makeResolverURL(URL, addr),
|
2022-09-01 15:45:42 +02:00
|
|
|
Timeout: 15 * time.Second,
|
2021-02-02 12:05:47 +01:00
|
|
|
},
|
|
|
|
Target: fmt.Sprintf("dnslookup://%s", domain), // urlgetter wants a URL
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// 7. make sure we don't test the same endpoint too frequently
|
|
|
|
// because this may cause residual censorship.
|
|
|
|
for _, input := range inputs {
|
|
|
|
resolverURL := input.Config.ResolverURL
|
|
|
|
m.Endpoints.maybeSleep(resolverURL, sess.Logger())
|
|
|
|
}
|
|
|
|
|
|
|
|
// 8. perform all the required resolutions
|
2022-06-02 09:31:52 +02:00
|
|
|
for output := range Collect(ctx, multi, inputs, sess.Logger()) {
|
2021-02-02 12:05:47 +01:00
|
|
|
resolverURL := output.Input.Config.ResolverURL
|
|
|
|
tk.Lookups[resolverURL] = output.TestKeys
|
|
|
|
m.Endpoints.maybeRegister(resolverURL)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-01-07 18:33:37 +01:00
|
|
|
func (m *Measurer) lookupHost(ctx context.Context, hostname string, r model.Resolver) ([]string, error) {
|
2022-09-01 15:45:42 +02:00
|
|
|
ctx, cancel := context.WithTimeout(ctx, 10*time.Second)
|
2021-02-02 12:05:47 +01:00
|
|
|
defer cancel()
|
|
|
|
return r.LookupHost(ctx, hostname)
|
|
|
|
}
|
|
|
|
|
|
|
|
// httpHost returns the configured HTTP host, if set, otherwise
|
|
|
|
// it will return the host provide as argument.
|
|
|
|
func (m *Measurer) httpHost(httpHost string) string {
|
|
|
|
if m.Config.HTTPHost != "" {
|
|
|
|
return m.Config.HTTPHost
|
|
|
|
}
|
|
|
|
return httpHost
|
|
|
|
}
|
|
|
|
|
|
|
|
// tlsServerName is like httpHost for the TLS server name.
|
|
|
|
func (m *Measurer) tlsServerName(tlsServerName string) string {
|
|
|
|
if m.Config.TLSServerName != "" {
|
|
|
|
return m.Config.TLSServerName
|
|
|
|
}
|
|
|
|
return tlsServerName
|
|
|
|
}
|
|
|
|
|
|
|
|
// Collect prints on the output channel the result of running dnscheck
|
|
|
|
// on every provided input. It closes the output channel when done.
|
|
|
|
func Collect(ctx context.Context, multi urlgetter.Multi, inputs []urlgetter.MultiInput,
|
2022-06-02 09:31:52 +02:00
|
|
|
logger model.Logger) <-chan urlgetter.MultiOutput {
|
2021-02-02 12:05:47 +01:00
|
|
|
outputch := make(chan urlgetter.MultiOutput)
|
|
|
|
expect := len(inputs)
|
|
|
|
inputch := multi.Run(ctx, inputs)
|
|
|
|
go func() {
|
|
|
|
var count int
|
|
|
|
defer close(outputch)
|
|
|
|
for count < expect {
|
|
|
|
entry := <-inputch
|
|
|
|
count++
|
2022-06-02 09:31:52 +02:00
|
|
|
logger.Infof("dnscheck: measure %s: %+v", entry.Input.Config.ResolverURL,
|
|
|
|
model.ErrorToStringOrOK(entry.Err))
|
2021-02-02 12:05:47 +01:00
|
|
|
outputch <- entry
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
return outputch
|
|
|
|
}
|
|
|
|
|
|
|
|
// makeResolverURL rewrites the input URL to replace the domain in
|
|
|
|
// the input URL with the given addr. When the input URL already contains
|
|
|
|
// an addr, this operation will return the same URL.
|
|
|
|
func makeResolverURL(URL *url.URL, addr string) string {
|
|
|
|
// 1. determine the hostname in the resulting URL
|
|
|
|
hostname := URL.Hostname()
|
|
|
|
if net.ParseIP(hostname) == nil {
|
|
|
|
hostname = addr
|
|
|
|
}
|
|
|
|
// 2. adjust hostname if we also have a port
|
|
|
|
if hasPort := URL.Port() != ""; hasPort {
|
|
|
|
_, port, err := net.SplitHostPort(URL.Host)
|
|
|
|
// We say this cannot fail because we already parsed the URL to validate
|
|
|
|
// its scheme and hence the URL hostname should be well formed.
|
|
|
|
runtimex.PanicOnError(err, "net.SplitHostPort should not fail here")
|
|
|
|
hostname = net.JoinHostPort(hostname, port)
|
|
|
|
} else if idx := strings.Index(addr, ":"); idx >= 0 {
|
|
|
|
// Make sure an IPv6 address hostname without a port is properly
|
|
|
|
// quoted to avoid breaking the URL parser down the line.
|
|
|
|
hostname = "[" + addr + "]"
|
|
|
|
}
|
|
|
|
// 3. reassemble the URL
|
|
|
|
return (&url.URL{
|
|
|
|
Scheme: URL.Scheme,
|
|
|
|
Host: hostname,
|
|
|
|
Path: URL.Path,
|
|
|
|
RawQuery: URL.RawQuery,
|
|
|
|
}).String()
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewExperimentMeasurer creates a new ExperimentMeasurer.
|
|
|
|
func NewExperimentMeasurer(config Config) model.ExperimentMeasurer {
|
2022-09-02 13:31:24 +02:00
|
|
|
return &Measurer{
|
|
|
|
Config: config,
|
|
|
|
Endpoints: nil, // disabled by default
|
|
|
|
}
|
2021-02-02 12:05:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// SummaryKeys contains summary keys for this experiment.
|
|
|
|
//
|
2022-05-09 09:33:18 +02:00
|
|
|
// Note that this structure is part of the ABI contract with ooniprobe
|
2021-02-02 12:05:47 +01:00
|
|
|
// therefore we should be careful when changing it.
|
|
|
|
type SummaryKeys struct {
|
|
|
|
IsAnomaly bool `json:"-"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetSummaryKeys implements model.ExperimentMeasurer.GetSummaryKeys.
|
|
|
|
func (m *Measurer) GetSummaryKeys(measurement *model.Measurement) (interface{}, error) {
|
|
|
|
return SummaryKeys{IsAnomaly: false}, nil
|
|
|
|
}
|