0ef1f24617
See https://github.com/ooni/probe/issues/2231 Co-authored-by: decfox <decfox@github.com> Co-authored-by: Simone Basso <bassosimone@gmail.com>
160 lines
3.8 KiB
Go
160 lines
3.8 KiB
Go
package dnsping
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"log"
|
|
"net"
|
|
"net/url"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/miekg/dns"
|
|
"github.com/ooni/probe-cli/v3/internal/engine/mockable"
|
|
"github.com/ooni/probe-cli/v3/internal/model"
|
|
"github.com/ooni/probe-cli/v3/internal/runtimex"
|
|
)
|
|
|
|
func TestConfig_domains(t *testing.T) {
|
|
c := Config{}
|
|
if c.domains() != "edge-chat.instagram.com example.com" {
|
|
t.Fatal("invalid default domains list")
|
|
}
|
|
}
|
|
|
|
func TestConfig_repetitions(t *testing.T) {
|
|
c := Config{}
|
|
if c.repetitions() != 10 {
|
|
t.Fatal("invalid default number of repetitions")
|
|
}
|
|
}
|
|
|
|
func TestConfig_delay(t *testing.T) {
|
|
c := Config{}
|
|
if c.delay() != time.Second {
|
|
t.Fatal("invalid default delay")
|
|
}
|
|
}
|
|
|
|
func TestMeasurer_run(t *testing.T) {
|
|
// expectedPings is the expected number of pings
|
|
const expectedPings = 4
|
|
|
|
// runHelper is an helper function to run this set of tests.
|
|
runHelper := func(input string) (*model.Measurement, model.ExperimentMeasurer, error) {
|
|
m := NewExperimentMeasurer(Config{
|
|
Domains: "example.com",
|
|
Delay: 1, // millisecond
|
|
Repetitions: expectedPings,
|
|
})
|
|
if m.ExperimentName() != "dnsping" {
|
|
t.Fatal("invalid experiment name")
|
|
}
|
|
if m.ExperimentVersion() != "0.3.0" {
|
|
t.Fatal("invalid experiment version")
|
|
}
|
|
ctx := context.Background()
|
|
meas := &model.Measurement{
|
|
Input: model.MeasurementTarget(input),
|
|
}
|
|
sess := &mockable.Session{
|
|
MockableLogger: model.DiscardLogger,
|
|
}
|
|
callbacks := model.NewPrinterCallbacks(model.DiscardLogger)
|
|
err := m.Run(ctx, sess, meas, callbacks)
|
|
return meas, m, err
|
|
}
|
|
|
|
t.Run("with empty input", func(t *testing.T) {
|
|
_, _, err := runHelper("")
|
|
if !errors.Is(err, errNoInputProvided) {
|
|
t.Fatal("unexpected error", err)
|
|
}
|
|
})
|
|
|
|
t.Run("with invalid URL", func(t *testing.T) {
|
|
_, _, err := runHelper("\t")
|
|
if !errors.Is(err, errInputIsNotAnURL) {
|
|
t.Fatal("unexpected error", err)
|
|
}
|
|
})
|
|
|
|
t.Run("with invalid scheme", func(t *testing.T) {
|
|
_, _, err := runHelper("https://8.8.8.8:443/")
|
|
if !errors.Is(err, errInvalidScheme) {
|
|
t.Fatal("unexpected error", err)
|
|
}
|
|
})
|
|
|
|
t.Run("with missing port", func(t *testing.T) {
|
|
_, _, err := runHelper("udp://8.8.8.8")
|
|
if !errors.Is(err, errMissingPort) {
|
|
t.Fatal("unexpected error", err)
|
|
}
|
|
})
|
|
|
|
t.Run("with local listener", func(t *testing.T) {
|
|
srvrURL, dnsListener, err := startDNSServer()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
defer dnsListener.Close()
|
|
meas, m, err := runHelper(srvrURL)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
tk := meas.TestKeys.(*TestKeys)
|
|
if len(tk.Pings) != expectedPings*2 { // account for A & AAAA pings
|
|
t.Fatal("unexpected number of pings")
|
|
}
|
|
ask, err := m.GetSummaryKeys(meas)
|
|
if err != nil {
|
|
t.Fatal("cannot obtain summary")
|
|
}
|
|
summary := ask.(SummaryKeys)
|
|
if summary.IsAnomaly {
|
|
t.Fatal("expected no anomaly")
|
|
}
|
|
})
|
|
}
|
|
|
|
// startDNSServer starts a local DNS server.
|
|
func startDNSServer() (string, net.PacketConn, error) {
|
|
dnsListener, err := net.ListenPacket("udp", "127.0.0.1:0")
|
|
if err != nil {
|
|
return "", nil, err
|
|
}
|
|
go runDNSServer(dnsListener)
|
|
URL := &url.URL{
|
|
Scheme: "udp",
|
|
Host: dnsListener.LocalAddr().String(),
|
|
Path: "/",
|
|
}
|
|
return URL.String(), dnsListener, nil
|
|
}
|
|
|
|
// runDNSServer runs the DNS server.
|
|
func runDNSServer(dnsListener net.PacketConn) {
|
|
ds := &dns.Server{
|
|
Handler: &dnsHandler{},
|
|
Net: "udp",
|
|
PacketConn: dnsListener,
|
|
}
|
|
err := ds.ActivateAndServe()
|
|
if !errors.Is(err, net.ErrClosed) {
|
|
runtimex.PanicOnError(err, "ActivateAndServe failed")
|
|
}
|
|
}
|
|
|
|
// dnsHandler handles DNS requests.
|
|
type dnsHandler struct{}
|
|
|
|
// ServeDNS serves a DNS request
|
|
func (h *dnsHandler) ServeDNS(rw dns.ResponseWriter, req *dns.Msg) {
|
|
m := new(dns.Msg)
|
|
m.Compress = true
|
|
m.MsgHdr.RecursionAvailable = true
|
|
m.SetRcode(req, dns.RcodeServerFailure)
|
|
rw.WriteMsg(m)
|
|
}
|