2022-07-05 19:10:39 +02:00
|
|
|
package main
|
2021-02-02 12:05:47 +01:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"net/http"
|
|
|
|
"sync"
|
|
|
|
"testing"
|
2021-10-13 16:37:02 +02:00
|
|
|
|
|
|
|
"github.com/google/go-cmp/cmp"
|
2022-07-05 18:41:35 +02:00
|
|
|
"github.com/ooni/probe-cli/v3/internal/model"
|
2022-07-05 20:25:18 +02:00
|
|
|
"github.com/ooni/probe-cli/v3/internal/model/mocks"
|
2021-10-13 16:37:02 +02:00
|
|
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
2021-02-02 12:05:47 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestHTTPDoWithInvalidURL(t *testing.T) {
|
|
|
|
ctx := context.Background()
|
|
|
|
wg := new(sync.WaitGroup)
|
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: nil,
|
|
|
|
MaxAcceptableBody: 1 << 24,
|
2022-07-05 18:41:35 +02:00
|
|
|
NewClient: func() model.HTTPClient {
|
|
|
|
return http.DefaultClient
|
|
|
|
},
|
|
|
|
Out: httpch,
|
|
|
|
URL: "http://[::1]aaaa",
|
|
|
|
Wg: wg,
|
2021-02-02 12:05:47 +01:00
|
|
|
})
|
|
|
|
// wait for measurement steps to complete
|
|
|
|
wg.Wait()
|
|
|
|
resp := <-httpch
|
2021-10-13 16:37:02 +02:00
|
|
|
if resp.Failure == nil || *resp.Failure != "unknown_error" {
|
2021-02-02 12:05:47 +01:00
|
|
|
t.Fatal("not the failure we expected")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestHTTPDoWithHTTPTransportFailure(t *testing.T) {
|
|
|
|
expected := errors.New("mocked error")
|
|
|
|
ctx := context.Background()
|
|
|
|
wg := new(sync.WaitGroup)
|
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: nil,
|
|
|
|
MaxAcceptableBody: 1 << 24,
|
2022-07-05 18:41:35 +02:00
|
|
|
NewClient: func() model.HTTPClient {
|
|
|
|
return &http.Client{
|
2022-07-05 20:25:18 +02:00
|
|
|
Transport: &mocks.HTTPTransport{
|
|
|
|
MockRoundTrip: func(req *http.Request) (*http.Response, error) {
|
|
|
|
return nil, expected
|
|
|
|
},
|
|
|
|
MockCloseIdleConnections: func() {
|
|
|
|
// nothing
|
|
|
|
},
|
2022-07-05 18:41:35 +02:00
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Out: httpch,
|
|
|
|
URL: "http://www.x.org",
|
|
|
|
Wg: wg,
|
2021-02-02 12:05:47 +01:00
|
|
|
})
|
|
|
|
// wait for measurement steps to complete
|
|
|
|
wg.Wait()
|
|
|
|
resp := <-httpch
|
2021-10-13 16:37:02 +02:00
|
|
|
if resp.Failure == nil || *resp.Failure != "unknown_error" {
|
2021-02-02 12:05:47 +01:00
|
|
|
t.Fatal("not the error we expected")
|
|
|
|
}
|
|
|
|
}
|
2021-10-13 16:37:02 +02:00
|
|
|
|
|
|
|
func newErrWrapper(failure, operation string) error {
|
|
|
|
return &netxlite.ErrWrapper{
|
|
|
|
Failure: failure,
|
|
|
|
Operation: operation,
|
|
|
|
WrappedErr: nil, // should not matter
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func newErrWrapperTopLevel(failure string) error {
|
|
|
|
return newErrWrapper(failure, netxlite.TopLevelOperation)
|
|
|
|
}
|
|
|
|
|
|
|
|
func Test_httpMapFailure(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
failure error
|
|
|
|
want *string
|
|
|
|
}{{
|
|
|
|
name: "nil",
|
|
|
|
failure: nil,
|
|
|
|
want: nil,
|
|
|
|
}, {
|
|
|
|
name: "nxdomain",
|
|
|
|
failure: newErrWrapperTopLevel(netxlite.FailureDNSNXDOMAINError),
|
|
|
|
want: stringPointerForString("dns_lookup_error"),
|
|
|
|
}, {
|
|
|
|
name: "no answer",
|
|
|
|
failure: newErrWrapperTopLevel(netxlite.FailureDNSNoAnswer),
|
|
|
|
want: stringPointerForString("dns_lookup_error"),
|
|
|
|
}, {
|
|
|
|
name: "non recoverable failure",
|
|
|
|
failure: newErrWrapperTopLevel(netxlite.FailureDNSNonRecoverableFailure),
|
|
|
|
want: stringPointerForString("dns_lookup_error"),
|
|
|
|
}, {
|
|
|
|
name: "refused",
|
|
|
|
failure: newErrWrapperTopLevel(netxlite.FailureDNSRefusedError),
|
|
|
|
want: stringPointerForString("dns_lookup_error"),
|
|
|
|
}, {
|
|
|
|
name: "server misbehaving",
|
|
|
|
failure: newErrWrapperTopLevel(netxlite.FailureDNSServerMisbehaving),
|
|
|
|
want: stringPointerForString("dns_lookup_error"),
|
|
|
|
}, {
|
|
|
|
name: "temporary failure",
|
|
|
|
failure: newErrWrapperTopLevel(netxlite.FailureDNSTemporaryFailure),
|
|
|
|
want: stringPointerForString("dns_lookup_error"),
|
|
|
|
}, {
|
|
|
|
name: "timeout outside of dns lookup",
|
|
|
|
failure: newErrWrapperTopLevel(netxlite.FailureGenericTimeoutError),
|
|
|
|
want: stringPointerForString(netxlite.FailureGenericTimeoutError),
|
|
|
|
}, {
|
|
|
|
name: "timeout inside of dns lookup",
|
|
|
|
failure: newErrWrapper(netxlite.FailureGenericTimeoutError, netxlite.ResolveOperation),
|
|
|
|
want: stringPointerForString("dns_lookup_error"),
|
|
|
|
}, {
|
|
|
|
name: "connection refused",
|
|
|
|
failure: newErrWrapperTopLevel(netxlite.FailureConnectionRefused),
|
|
|
|
want: stringPointerForString("connection_refused_error"),
|
|
|
|
}, {
|
|
|
|
name: "anything else",
|
|
|
|
failure: newErrWrapperTopLevel(netxlite.FailureEOFError),
|
|
|
|
want: stringPointerForString("unknown_error"),
|
|
|
|
}}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
got := httpMapFailure(tt.failure)
|
|
|
|
if diff := cmp.Diff(tt.want, got); diff != "" {
|
|
|
|
t.Fatal(diff)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|