88236a4352
This experiment pings a QUIC-able host. It can be used to measure QUIC availability independently from TLS. This is the reference issue: https://github.com/ooni/probe/issues/1994 ### A QUIC PING is: - a QUIC Initial packet with a size of 1200 bytes (minimum datagram size defined in the [RFC 9000](https://www.rfc-editor.org/rfc/rfc9000.html#initial-size)), - with a random payload (i.e. no TLS ClientHello), - with the version string 0xbabababa which forces Version Negotiation at the server. QUIC-able hosts respond to the QUIC PING with a Version Negotiation packet. The input is a domain name or an IP address. The default port used by quicping is 443, as this is the port used by HTTP/3. The port can be modified with the `-O Port=` option. The default number of repetitions is 10, it can be changed with `-O Repetitions=`. ### Usage: ``` ./miniooni -i google.com quicping ./miniooni -i 142.250.181.206 quicping ./miniooni -i 142.250.181.206 -OPort=443 quicping ./miniooni -i 142.250.181.206 -ORepetitions=2 quicping ```
311 lines
9.0 KiB
Go
311 lines
9.0 KiB
Go
package quicping
|
|
|
|
import (
|
|
"context"
|
|
"encoding/hex"
|
|
"errors"
|
|
"net"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/apex/log"
|
|
"github.com/ooni/probe-cli/v3/internal/engine/mockable"
|
|
"github.com/ooni/probe-cli/v3/internal/model"
|
|
"github.com/ooni/probe-cli/v3/internal/model/mocks"
|
|
)
|
|
|
|
// FailStdLib is a failing model.UnderlyingNetworkLibrary.
|
|
type FailStdLib struct {
|
|
conn model.UDPLikeConn
|
|
err error
|
|
writeErr error
|
|
readErr error
|
|
}
|
|
|
|
// ListenUDP implements UnderlyingNetworkLibrary.ListenUDP.
|
|
func (f *FailStdLib) ListenUDP(network string, laddr *net.UDPAddr) (model.UDPLikeConn, error) {
|
|
conn, _ := net.ListenUDP(network, laddr)
|
|
f.conn = model.UDPLikeConn(conn)
|
|
if f.err != nil {
|
|
return nil, f.err
|
|
}
|
|
if f.writeErr != nil {
|
|
return &mocks.UDPLikeConn{
|
|
MockWriteTo: func(p []byte, addr net.Addr) (int, error) {
|
|
return 0, f.writeErr
|
|
},
|
|
MockReadFrom: func(p []byte) (int, net.Addr, error) {
|
|
return f.conn.ReadFrom(p)
|
|
},
|
|
MockSetDeadline: func(t time.Time) error {
|
|
return f.conn.SetDeadline(t)
|
|
},
|
|
MockClose: func() error {
|
|
return f.conn.Close()
|
|
},
|
|
}, nil
|
|
}
|
|
if f.readErr != nil {
|
|
return &mocks.UDPLikeConn{
|
|
MockWriteTo: func(p []byte, addr net.Addr) (int, error) {
|
|
return f.conn.WriteTo(p, addr)
|
|
},
|
|
MockReadFrom: func(p []byte) (int, net.Addr, error) {
|
|
return 0, nil, f.readErr
|
|
},
|
|
MockSetDeadline: func(t time.Time) error {
|
|
return f.conn.SetDeadline(t)
|
|
},
|
|
MockClose: func() error {
|
|
return f.conn.Close()
|
|
},
|
|
}, nil
|
|
}
|
|
return &mocks.UDPLikeConn{}, nil
|
|
}
|
|
|
|
// LookupHost implements UnderlyingNetworkLibrary.LookupHost.
|
|
func (f *FailStdLib) LookupHost(ctx context.Context, domain string) ([]string, error) {
|
|
return nil, f.err
|
|
}
|
|
|
|
// NewSimpleDialer implements UnderlyingNetworkLibrary.NewSimpleDialer.
|
|
func (f *FailStdLib) NewSimpleDialer(timeout time.Duration) model.SimpleDialer {
|
|
return nil
|
|
}
|
|
|
|
func TestNewExperimentMeasurer(t *testing.T) {
|
|
measurer := NewExperimentMeasurer(Config{})
|
|
if measurer.ExperimentName() != "quicping" {
|
|
t.Fatal("unexpected name")
|
|
}
|
|
if measurer.ExperimentVersion() != "0.1.0" {
|
|
t.Fatal("unexpected version")
|
|
}
|
|
}
|
|
|
|
func TestInvalidHost(t *testing.T) {
|
|
measurer := NewExperimentMeasurer(Config{
|
|
Port: 443,
|
|
Repetitions: 1,
|
|
})
|
|
measurement := new(model.Measurement)
|
|
measurement.Input = model.MeasurementTarget("a.a.a.a")
|
|
sess := &mockable.Session{MockableLogger: log.Log}
|
|
err := measurer.Run(context.Background(), sess, measurement,
|
|
model.NewPrinterCallbacks(log.Log))
|
|
if err == nil {
|
|
t.Fatal("expected an error here")
|
|
}
|
|
if _, ok := err.(*net.DNSError); !ok {
|
|
t.Fatal("unexpected error type")
|
|
}
|
|
}
|
|
|
|
func TestURLInput(t *testing.T) {
|
|
measurer := NewExperimentMeasurer(Config{
|
|
Repetitions: 1,
|
|
})
|
|
measurement := new(model.Measurement)
|
|
measurement.Input = model.MeasurementTarget("https://google.com/")
|
|
sess := &mockable.Session{MockableLogger: log.Log}
|
|
err := measurer.Run(context.Background(), sess, measurement,
|
|
model.NewPrinterCallbacks(log.Log))
|
|
if err != nil {
|
|
t.Fatal("unexpected error")
|
|
}
|
|
tk := measurement.TestKeys.(*TestKeys)
|
|
if tk.Domain != "google.com" {
|
|
t.Fatal("unexpected domain")
|
|
}
|
|
|
|
}
|
|
|
|
func TestSuccess(t *testing.T) {
|
|
measurer := NewExperimentMeasurer(Config{})
|
|
measurement := new(model.Measurement)
|
|
measurement.Input = model.MeasurementTarget("google.com")
|
|
sess := &mockable.Session{MockableLogger: log.Log}
|
|
err := measurer.Run(context.Background(), sess, measurement,
|
|
model.NewPrinterCallbacks(log.Log))
|
|
if err != nil {
|
|
t.Fatal("did not expect an error here")
|
|
}
|
|
tk := measurement.TestKeys.(*TestKeys)
|
|
if tk.Domain != "google.com" {
|
|
t.Fatal("unexpected domain")
|
|
}
|
|
if tk.Repetitions != 10 {
|
|
t.Fatal("unexpected number of repetitions, default is 10")
|
|
}
|
|
if tk.Pings == nil || len(tk.Pings) != 10 {
|
|
t.Fatal("unexpected number of pings", len(tk.Pings))
|
|
}
|
|
for i, ping := range tk.Pings {
|
|
if ping.Failure != nil {
|
|
t.Fatal("ping failed unexpectedly", i, *ping.Failure)
|
|
}
|
|
for _, resp := range ping.Responses {
|
|
if resp.Failure != nil {
|
|
t.Fatal("unexepcted response failure")
|
|
}
|
|
if resp.SupportedVersions == nil || len(resp.SupportedVersions) == 0 {
|
|
t.Fatal("server did not respond with supported versions")
|
|
}
|
|
}
|
|
}
|
|
sk, err := measurer.GetSummaryKeys(measurement)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if _, ok := sk.(SummaryKeys); !ok {
|
|
t.Fatal("invalid type for summary keys")
|
|
}
|
|
}
|
|
|
|
func TestWithCancelledContext(t *testing.T) {
|
|
measurer := NewExperimentMeasurer(Config{})
|
|
measurement := new(model.Measurement)
|
|
measurement.Input = model.MeasurementTarget("google.com")
|
|
sess := &mockable.Session{MockableLogger: log.Log}
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
cancel()
|
|
err := measurer.Run(ctx, sess, measurement,
|
|
model.NewPrinterCallbacks(log.Log))
|
|
if err != nil {
|
|
t.Fatal("did not expect an error here")
|
|
}
|
|
tk := measurement.TestKeys.(*TestKeys)
|
|
if len(tk.Pings) > 0 {
|
|
t.Fatal("there should not be any measurements")
|
|
}
|
|
}
|
|
|
|
func TestListenFails(t *testing.T) {
|
|
expected := errors.New("expected")
|
|
measurer := NewExperimentMeasurer(Config{
|
|
networkLib: &FailStdLib{err: expected, readErr: nil, writeErr: nil},
|
|
})
|
|
measurement := new(model.Measurement)
|
|
measurement.Input = model.MeasurementTarget("google.com")
|
|
sess := &mockable.Session{MockableLogger: log.Log}
|
|
err := measurer.Run(context.Background(), sess, measurement,
|
|
model.NewPrinterCallbacks(log.Log))
|
|
if err == nil {
|
|
t.Fatal("expected an error here")
|
|
}
|
|
if err != expected {
|
|
t.Fatal("unexpected error type")
|
|
}
|
|
}
|
|
|
|
func TestWriteFails(t *testing.T) {
|
|
expected := errors.New("expected")
|
|
measurer := NewExperimentMeasurer(Config{
|
|
networkLib: &FailStdLib{err: nil, readErr: nil, writeErr: expected},
|
|
Repetitions: 1,
|
|
})
|
|
measurement := new(model.Measurement)
|
|
measurement.Input = model.MeasurementTarget("google.com")
|
|
sess := &mockable.Session{MockableLogger: log.Log}
|
|
err := measurer.Run(context.Background(), sess, measurement,
|
|
model.NewPrinterCallbacks(log.Log))
|
|
if err != nil {
|
|
t.Fatal("unexpected error")
|
|
}
|
|
tk := measurement.TestKeys.(*TestKeys)
|
|
if tk.Pings == nil || len(tk.Pings) != 1 {
|
|
t.Fatal("unexpected number of pings", len(tk.Pings))
|
|
}
|
|
for i, ping := range tk.Pings {
|
|
if ping.Failure == nil {
|
|
t.Fatal("expected an error here, ping", i)
|
|
}
|
|
if !strings.Contains(*ping.Failure, "expected") {
|
|
t.Fatal("ping: unexpected error type", i, *ping.Failure)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestReadFails(t *testing.T) {
|
|
expected := errors.New("expected")
|
|
measurer := NewExperimentMeasurer(Config{
|
|
networkLib: &FailStdLib{err: nil, readErr: expected, writeErr: nil},
|
|
Repetitions: 1,
|
|
})
|
|
measurement := new(model.Measurement)
|
|
measurement.Input = model.MeasurementTarget("google.com")
|
|
sess := &mockable.Session{MockableLogger: log.Log}
|
|
err := measurer.Run(context.Background(), sess, measurement,
|
|
model.NewPrinterCallbacks(log.Log))
|
|
if err != nil {
|
|
t.Fatal("unexpected error")
|
|
}
|
|
tk := measurement.TestKeys.(*TestKeys)
|
|
if tk.Pings == nil || len(tk.Pings) != 1 {
|
|
t.Fatal("unexpected number of pings", len(tk.Pings))
|
|
}
|
|
for i, ping := range tk.Pings {
|
|
if ping.Failure == nil {
|
|
t.Fatal("expected an error here, ping", i)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestNoResponse(t *testing.T) {
|
|
measurer := NewExperimentMeasurer(Config{
|
|
Repetitions: 1,
|
|
})
|
|
measurement := new(model.Measurement)
|
|
measurement.Input = model.MeasurementTarget("ooni.org")
|
|
sess := &mockable.Session{MockableLogger: log.Log}
|
|
err := measurer.Run(context.Background(), sess, measurement,
|
|
model.NewPrinterCallbacks(log.Log))
|
|
if err != nil {
|
|
t.Fatal("did not expect an error here")
|
|
}
|
|
tk := measurement.TestKeys.(*TestKeys)
|
|
if tk.Pings == nil || len(tk.Pings) != 1 {
|
|
t.Fatal("unexpected number of pings", len(tk.Pings))
|
|
}
|
|
if tk.Pings[0].Failure == nil {
|
|
t.Fatal("expected an error here")
|
|
}
|
|
if *tk.Pings[0].Failure != "generic_timeout_error" {
|
|
t.Fatal("unexpected error type")
|
|
}
|
|
}
|
|
|
|
func TestDissect(t *testing.T) {
|
|
// destID--srcID: 040b9649d3fd4c038ab6c073966f3921--44d064031288e97646451f
|
|
versionNegotiationResponse, _ := hex.DecodeString("eb0000000010040b9649d3fd4c038ab6c073966f39210b44d064031288e97646451f00000001ff00001dff00001cff00001b")
|
|
measurer := NewExperimentMeasurer(Config{})
|
|
destID := "040b9649d3fd4c038ab6c073966f3921"
|
|
_, dst, err := measurer.(*Measurer).dissectVersionNegotiation(versionNegotiationResponse)
|
|
if err != nil {
|
|
t.Fatal("unexpected error", err)
|
|
}
|
|
if hex.EncodeToString(dst) != destID {
|
|
t.Fatal("unexpected destination connection ID")
|
|
}
|
|
|
|
versionNegotiationResponse[1] = byte(0xff)
|
|
_, _, err = measurer.(*Measurer).dissectVersionNegotiation(versionNegotiationResponse)
|
|
if err == nil {
|
|
t.Fatal("expected an error here", err)
|
|
}
|
|
if !strings.HasSuffix(err.Error(), "unexpected Version Negotiation format") {
|
|
t.Fatal("unexpected error type", err)
|
|
}
|
|
|
|
versionNegotiationResponse[0] = byte(0x01)
|
|
_, _, err = measurer.(*Measurer).dissectVersionNegotiation(versionNegotiationResponse)
|
|
if err == nil {
|
|
t.Fatal("expected an error here", err)
|
|
}
|
|
if !strings.HasSuffix(err.Error(), "not a long header packet") {
|
|
t.Fatal("unexpected error type", err)
|
|
}
|
|
}
|