2022-07-05 19:10:39 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
//
|
|
|
|
// Top-level measurement algorithm
|
|
|
|
//
|
2021-02-02 12:05:47 +01:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"net"
|
|
|
|
"net/url"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/ooni/probe-cli/v3/internal/engine/experiment/webconnectivity"
|
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
2022-07-05 19:10:39 +02:00
|
|
|
// ctrlRequest is the request sent to the test helper
|
|
|
|
ctrlRequest = webconnectivity.ControlRequest
|
2021-02-02 12:05:47 +01:00
|
|
|
|
2022-07-05 19:10:39 +02:00
|
|
|
// ctrlResponse is the response from the test helper
|
|
|
|
ctrlResponse = webconnectivity.ControlResponse
|
2021-02-02 12:05:47 +01:00
|
|
|
)
|
|
|
|
|
2022-07-05 19:10:39 +02:00
|
|
|
// measure performs the measurement described by the request and
|
2021-02-02 12:05:47 +01:00
|
|
|
// returns the corresponding response or an error.
|
2022-07-05 19:10:39 +02:00
|
|
|
func measure(ctx context.Context, config *handler, creq *ctrlRequest) (*ctrlResponse, error) {
|
2021-02-02 12:05:47 +01:00
|
|
|
// parse input for correctness
|
|
|
|
URL, err := url.Parse(creq.HTTPRequest)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-07-05 19:10:39 +02:00
|
|
|
wg := &sync.WaitGroup{}
|
|
|
|
|
2021-02-02 12:05:47 +01:00
|
|
|
// dns: start
|
2022-07-05 19:10:39 +02:00
|
|
|
dnsch := make(chan ctrlDNSResult, 1)
|
2021-02-02 12:05:47 +01:00
|
|
|
if net.ParseIP(URL.Hostname()) == nil {
|
|
|
|
wg.Add(1)
|
2022-07-05 19:10:39 +02:00
|
|
|
go dnsDo(ctx, &dnsConfig{
|
2022-07-05 18:41:35 +02:00
|
|
|
Domain: URL.Hostname(),
|
|
|
|
NewResolver: config.NewResolver,
|
|
|
|
Out: dnsch,
|
|
|
|
Wg: wg,
|
2021-02-02 12:05:47 +01:00
|
|
|
})
|
|
|
|
}
|
2022-07-05 19:10:39 +02:00
|
|
|
|
2022-08-28 12:17:31 +02:00
|
|
|
// wait for DNS measurements to complete
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
// start assembling the response
|
|
|
|
cresp := new(ctrlResponse)
|
|
|
|
select {
|
|
|
|
case cresp.DNS = <-dnsch:
|
|
|
|
default:
|
|
|
|
// we need to emit a non-nil Addrs to match exactly
|
|
|
|
// the behavior of the legacy TH
|
|
|
|
cresp.DNS = ctrlDNSResult{
|
|
|
|
Failure: nil,
|
|
|
|
Addrs: []string{},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-02 12:05:47 +01:00
|
|
|
// tcpconnect: start
|
2022-07-05 19:10:39 +02:00
|
|
|
tcpconnch := make(chan tcpResultPair, len(creq.TCPConnect))
|
2021-02-02 12:05:47 +01:00
|
|
|
for _, endpoint := range creq.TCPConnect {
|
|
|
|
wg.Add(1)
|
2022-07-05 19:10:39 +02:00
|
|
|
go tcpDo(ctx, &tcpConfig{
|
2022-07-05 18:41:35 +02:00
|
|
|
Endpoint: endpoint,
|
|
|
|
NewDialer: config.NewDialer,
|
|
|
|
Out: tcpconnch,
|
|
|
|
Wg: wg,
|
2021-02-02 12:05:47 +01:00
|
|
|
})
|
|
|
|
}
|
2022-07-05 19:10:39 +02:00
|
|
|
|
2021-02-02 12:05:47 +01:00
|
|
|
// http: start
|
2022-07-05 19:10:39 +02:00
|
|
|
httpch := make(chan ctrlHTTPResponse, 1)
|
2021-02-02 12:05:47 +01:00
|
|
|
wg.Add(1)
|
2022-07-05 19:10:39 +02:00
|
|
|
go httpDo(ctx, &httpConfig{
|
2021-02-02 12:05:47 +01:00
|
|
|
Headers: creq.HTTPRequestHeaders,
|
|
|
|
MaxAcceptableBody: config.MaxAcceptableBody,
|
2022-07-05 18:41:35 +02:00
|
|
|
NewClient: config.NewClient,
|
2021-02-02 12:05:47 +01:00
|
|
|
Out: httpch,
|
|
|
|
URL: creq.HTTPRequest,
|
|
|
|
Wg: wg,
|
|
|
|
})
|
2022-07-05 19:10:39 +02:00
|
|
|
|
2022-08-28 12:17:31 +02:00
|
|
|
// wait for endpoint measurements to complete
|
2021-02-02 12:05:47 +01:00
|
|
|
wg.Wait()
|
2022-07-05 19:10:39 +02:00
|
|
|
|
2022-08-28 12:17:31 +02:00
|
|
|
// continue assembling the response
|
2021-02-02 12:05:47 +01:00
|
|
|
cresp.HTTPRequest = <-httpch
|
2022-07-05 19:10:39 +02:00
|
|
|
cresp.TCPConnect = make(map[string]ctrlTCPResult)
|
2022-08-28 12:17:31 +02:00
|
|
|
Loop:
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case tcpconn := <-tcpconnch:
|
|
|
|
cresp.TCPConnect[tcpconn.Endpoint] = tcpconn.Result
|
|
|
|
default:
|
|
|
|
break Loop
|
|
|
|
}
|
2021-02-02 12:05:47 +01:00
|
|
|
}
|
2022-07-05 19:10:39 +02:00
|
|
|
|
2021-02-02 12:05:47 +01:00
|
|
|
return cresp, nil
|
|
|
|
}
|