1eb9e8c9b0
While there, also change to pointer receiver and use internal testing for what are clearly unit tests. Part of https://github.com/ooni/probe/issues/1591.
195 lines
5.1 KiB
Go
195 lines
5.1 KiB
Go
package resolver_test
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"errors"
|
|
"io"
|
|
"net/http"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/miekg/dns"
|
|
"github.com/ooni/probe-cli/v3/internal/engine/legacy/netx/handlers"
|
|
"github.com/ooni/probe-cli/v3/internal/engine/legacy/netx/modelx"
|
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/resolver"
|
|
"github.com/ooni/probe-cli/v3/internal/netxlite/mocks"
|
|
)
|
|
|
|
func TestEmitterTransportSuccess(t *testing.T) {
|
|
ctx := context.Background()
|
|
handler := &handlers.SavingHandler{}
|
|
root := &modelx.MeasurementRoot{
|
|
Beginning: time.Now(),
|
|
Handler: handler,
|
|
}
|
|
ctx = modelx.WithMeasurementRoot(ctx, root)
|
|
txp := resolver.EmitterTransport{RoundTripper: resolver.FakeTransport{
|
|
Data: resolver.GenReplySuccess(t, dns.TypeA, "8.8.8.8"),
|
|
}}
|
|
e := resolver.MiekgEncoder{}
|
|
querydata, err := e.Encode("www.google.com", dns.TypeAAAA, true)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
replydata, err := txp.RoundTrip(ctx, querydata)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
events := handler.Read()
|
|
if len(events) != 2 {
|
|
t.Fatal("unexpected number of events")
|
|
}
|
|
if events[0].DNSQuery == nil {
|
|
t.Fatal("missing DNSQuery field")
|
|
}
|
|
if !bytes.Equal(events[0].DNSQuery.Data, querydata) {
|
|
t.Fatal("invalid query data")
|
|
}
|
|
if events[0].DNSQuery.DurationSinceBeginning <= 0 {
|
|
t.Fatal("invalid duration since beginning")
|
|
}
|
|
if events[1].DNSReply == nil {
|
|
t.Fatal("missing DNSReply field")
|
|
}
|
|
if !bytes.Equal(events[1].DNSReply.Data, replydata) {
|
|
t.Fatal("missing reply data")
|
|
}
|
|
if events[1].DNSReply.DurationSinceBeginning <= 0 {
|
|
t.Fatal("invalid duration since beginning")
|
|
}
|
|
}
|
|
|
|
func TestEmitterTransportFailure(t *testing.T) {
|
|
ctx := context.Background()
|
|
handler := &handlers.SavingHandler{}
|
|
root := &modelx.MeasurementRoot{
|
|
Beginning: time.Now(),
|
|
Handler: handler,
|
|
}
|
|
ctx = modelx.WithMeasurementRoot(ctx, root)
|
|
mocked := errors.New("mocked error")
|
|
txp := resolver.EmitterTransport{RoundTripper: resolver.FakeTransport{
|
|
Err: mocked,
|
|
}}
|
|
e := resolver.MiekgEncoder{}
|
|
querydata, err := e.Encode("www.google.com", dns.TypeAAAA, true)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
replydata, err := txp.RoundTrip(ctx, querydata)
|
|
if !errors.Is(err, mocked) {
|
|
t.Fatal("not the error we expected")
|
|
}
|
|
if replydata != nil {
|
|
t.Fatal("expected nil replydata")
|
|
}
|
|
events := handler.Read()
|
|
if len(events) != 1 {
|
|
t.Fatal("unexpected number of events")
|
|
}
|
|
if events[0].DNSQuery == nil {
|
|
t.Fatal("missing DNSQuery field")
|
|
}
|
|
if !bytes.Equal(events[0].DNSQuery.Data, querydata) {
|
|
t.Fatal("invalid query data")
|
|
}
|
|
if events[0].DNSQuery.DurationSinceBeginning <= 0 {
|
|
t.Fatal("invalid duration since beginning")
|
|
}
|
|
}
|
|
|
|
func TestEmitterResolverFailure(t *testing.T) {
|
|
ctx := context.Background()
|
|
handler := &handlers.SavingHandler{}
|
|
root := &modelx.MeasurementRoot{
|
|
Beginning: time.Now(),
|
|
Handler: handler,
|
|
}
|
|
ctx = modelx.WithMeasurementRoot(ctx, root)
|
|
r := resolver.EmitterResolver{Resolver: resolver.NewSerialResolver(
|
|
&resolver.DNSOverHTTPS{
|
|
Client: &mocks.HTTPClient{
|
|
MockDo: func(req *http.Request) (*http.Response, error) {
|
|
return nil, io.EOF
|
|
},
|
|
},
|
|
URL: "https://dns.google.com/",
|
|
},
|
|
)}
|
|
replies, err := r.LookupHost(ctx, "www.google.com")
|
|
if !errors.Is(err, io.EOF) {
|
|
t.Fatal("not the error we expected")
|
|
}
|
|
if replies != nil {
|
|
t.Fatal("expected nil replies")
|
|
}
|
|
events := handler.Read()
|
|
if len(events) != 2 {
|
|
t.Fatal("unexpected number of events")
|
|
}
|
|
if events[0].ResolveStart == nil {
|
|
t.Fatal("missing ResolveStart field")
|
|
}
|
|
if events[0].ResolveStart.DurationSinceBeginning <= 0 {
|
|
t.Fatal("invalid duration since beginning")
|
|
}
|
|
if events[0].ResolveStart.Hostname != "www.google.com" {
|
|
t.Fatal("invalid Hostname")
|
|
}
|
|
if events[0].ResolveStart.TransportAddress != "https://dns.google.com/" {
|
|
t.Fatal("invalid TransportAddress")
|
|
}
|
|
if events[0].ResolveStart.TransportNetwork != "doh" {
|
|
t.Fatal("invalid TransportNetwork")
|
|
}
|
|
if events[1].ResolveDone == nil {
|
|
t.Fatal("missing ResolveDone field")
|
|
}
|
|
if events[1].ResolveDone.DurationSinceBeginning <= 0 {
|
|
t.Fatal("invalid duration since beginning")
|
|
}
|
|
if events[1].ResolveDone.Error != io.EOF {
|
|
t.Fatal("invalid Error")
|
|
}
|
|
if events[1].ResolveDone.Hostname != "www.google.com" {
|
|
t.Fatal("invalid Hostname")
|
|
}
|
|
if events[1].ResolveDone.TransportAddress != "https://dns.google.com/" {
|
|
t.Fatal("invalid TransportAddress")
|
|
}
|
|
if events[1].ResolveDone.TransportNetwork != "doh" {
|
|
t.Fatal("invalid TransportNetwork")
|
|
}
|
|
}
|
|
|
|
func TestEmitterResolverSuccess(t *testing.T) {
|
|
ctx := context.Background()
|
|
handler := &handlers.SavingHandler{}
|
|
root := &modelx.MeasurementRoot{
|
|
Beginning: time.Now(),
|
|
Handler: handler,
|
|
}
|
|
ctx = modelx.WithMeasurementRoot(ctx, root)
|
|
r := resolver.EmitterResolver{Resolver: resolver.NewFakeResolverWithResult(
|
|
[]string{"8.8.8.8"},
|
|
)}
|
|
replies, err := r.LookupHost(ctx, "dns.google.com")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if len(replies) != 1 {
|
|
t.Fatal("expected a single replies")
|
|
}
|
|
events := handler.Read()
|
|
if len(events) != 2 {
|
|
t.Fatal("unexpected number of events")
|
|
}
|
|
if events[1].ResolveDone == nil {
|
|
t.Fatal("missing ResolveDone field")
|
|
}
|
|
if events[1].ResolveDone.Addresses[0] != "8.8.8.8" {
|
|
t.Fatal("invalid Addresses")
|
|
}
|
|
}
|