2022-07-01 12:22:22 +02:00
|
|
|
package measurexlite
|
|
|
|
|
|
|
|
//
|
|
|
|
// Definition of Trace
|
|
|
|
//
|
|
|
|
|
|
|
|
import (
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/ooni/probe-cli/v3/internal/model"
|
|
|
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Trace implements model.Trace.
|
|
|
|
//
|
|
|
|
// The zero-value of this struct is invalid. To construct you should either
|
|
|
|
// fill all the fields marked as MANDATORY or use NewTrace.
|
|
|
|
//
|
|
|
|
// Buffered channels
|
|
|
|
//
|
|
|
|
// NewTrace uses reasonable buffer sizes for the channels used for collecting
|
|
|
|
// events. You should drain the channels used by this implementation after
|
|
|
|
// each operation you perform (i.e., we expect you to peform step-by-step
|
|
|
|
// measurements). If you want larger (or smaller) buffers, then you should
|
|
|
|
// construct this data type manually with the desired buffer sizes.
|
|
|
|
//
|
|
|
|
// We have convenience methods for extracting events from the buffered
|
|
|
|
// channels. Otherwise, you could read the channels directly. (In which
|
|
|
|
// case, remember to issue nonblocking channel reads because channels are
|
|
|
|
// never closed and they're just written when new events occur.)
|
|
|
|
type Trace struct {
|
|
|
|
// Index is the MANDATORY unique index of this trace within the
|
|
|
|
// current measurement. If you don't care about uniquely identifying
|
2022-07-08 19:42:24 +02:00
|
|
|
// traces, you can use zero to indicate the "default" trace.
|
2022-07-01 12:22:22 +02:00
|
|
|
Index int64
|
|
|
|
|
|
|
|
// NetworkEvent is MANDATORY and buffers network events. If you create
|
|
|
|
// this channel manually, ensure it has some buffer.
|
|
|
|
NetworkEvent chan *model.ArchivalNetworkEvent
|
|
|
|
|
2022-08-11 18:18:25 +02:00
|
|
|
// NewStdlibResolverFn is OPTIONAL and can be used to overide
|
|
|
|
// calls to the netxlite.NewStdlibResolver factory.
|
|
|
|
NewStdlibResolverFn func(logger model.Logger) model.Resolver
|
|
|
|
|
2022-08-11 16:00:37 +02:00
|
|
|
// NewParallelUDPResolverFn is OPTIONAL and can be used to overide
|
|
|
|
// calls to the netxlite.NewParallelUDPResolver factory.
|
|
|
|
NewParallelUDPResolverFn func(logger model.Logger, dialer model.Dialer, address string) model.Resolver
|
|
|
|
|
|
|
|
// NewParallelDNSOverHTTPSResolverFn is OPTIONAL and can be used to overide
|
|
|
|
// calls to the netxlite.NewParallelDNSOverHTTPSUDPResolver factory.
|
|
|
|
NewParallelDNSOverHTTPSResolverFn func(logger model.Logger, URL string) model.Resolver
|
2022-07-08 19:42:24 +02:00
|
|
|
|
2022-07-01 12:22:22 +02:00
|
|
|
// NewDialerWithoutResolverFn is OPTIONAL and can be used to override
|
|
|
|
// calls to the netxlite.NewDialerWithoutResolver factory.
|
|
|
|
NewDialerWithoutResolverFn func(dl model.DebugLogger) model.Dialer
|
|
|
|
|
|
|
|
// NewTLSHandshakerStdlibFn is OPTIONAL and can be used to overide
|
|
|
|
// calls to the netxlite.NewTLSHandshakerStdlib factory.
|
|
|
|
NewTLSHandshakerStdlibFn func(dl model.DebugLogger) model.TLSHandshaker
|
|
|
|
|
2022-08-11 16:00:37 +02:00
|
|
|
// DNSLookup is MANDATORY and buffers DNS Lookup observations. If you create
|
|
|
|
// this channel manually, ensure it has some buffer.
|
|
|
|
DNSLookup chan *model.ArchivalDNSLookupResult
|
2022-07-08 19:42:24 +02:00
|
|
|
|
2022-07-01 12:22:22 +02:00
|
|
|
// TCPConnect is MANDATORY and buffers TCP connect observations. If you create
|
|
|
|
// this channel manually, ensure it has some buffer.
|
|
|
|
TCPConnect chan *model.ArchivalTCPConnectResult
|
|
|
|
|
|
|
|
// TLSHandshake is MANDATORY and buffers TLS handshake observations. If you create
|
|
|
|
// this channel manually, ensure it has some buffer.
|
|
|
|
TLSHandshake chan *model.ArchivalTLSOrQUICHandshakeResult
|
|
|
|
|
|
|
|
// TimeNowFn is OPTIONAL and can be used to override calls to time.Now
|
|
|
|
// to produce deterministic timing when testing.
|
|
|
|
TimeNowFn func() time.Time
|
|
|
|
|
|
|
|
// ZeroTime is the MANDATORY time when we started the current measurement.
|
|
|
|
ZeroTime time.Time
|
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
|
|
|
// NetworkEventBufferSize is the buffer size for constructing
|
|
|
|
// the Trace's NetworkEvent buffered channel.
|
|
|
|
NetworkEventBufferSize = 64
|
|
|
|
|
2022-07-08 19:42:24 +02:00
|
|
|
// DNSLookupBufferSize is the buffer size for constructing
|
|
|
|
// the Trace's DNSLookup map of buffered channels.
|
|
|
|
DNSLookupBufferSize = 8
|
|
|
|
|
2022-07-01 12:22:22 +02:00
|
|
|
// TCPConnectBufferSize is the buffer size for constructing
|
|
|
|
// the Trace's TCPConnect buffered channel.
|
|
|
|
TCPConnectBufferSize = 8
|
|
|
|
|
|
|
|
// TLSHandshakeBufferSize is the buffer for construcing
|
|
|
|
// the Trace's TLSHandshake buffered channel.
|
|
|
|
TLSHandshakeBufferSize = 8
|
|
|
|
)
|
|
|
|
|
|
|
|
// NewTrace creates a new instance of Trace using default settings.
|
|
|
|
//
|
|
|
|
// We create buffered channels using as buffer sizes the constants that
|
|
|
|
// are also defined by this package.
|
|
|
|
//
|
|
|
|
// Arguments:
|
|
|
|
//
|
|
|
|
// - index is the unique index of this trace within the current measurement (use
|
|
|
|
// zero if you don't care about giving this trace a unique ID);
|
|
|
|
//
|
|
|
|
// - zeroTime is the time when we started the current measurement.
|
|
|
|
func NewTrace(index int64, zeroTime time.Time) *Trace {
|
|
|
|
return &Trace{
|
|
|
|
Index: index,
|
|
|
|
NetworkEvent: make(
|
|
|
|
chan *model.ArchivalNetworkEvent,
|
|
|
|
NetworkEventBufferSize,
|
|
|
|
),
|
|
|
|
NewDialerWithoutResolverFn: nil, // use default
|
|
|
|
NewTLSHandshakerStdlibFn: nil, // use default
|
2022-08-11 16:00:37 +02:00
|
|
|
DNSLookup: make(
|
|
|
|
chan *model.ArchivalDNSLookupResult,
|
|
|
|
DNSLookupBufferSize,
|
|
|
|
),
|
2022-07-01 12:22:22 +02:00
|
|
|
TCPConnect: make(
|
|
|
|
chan *model.ArchivalTCPConnectResult,
|
|
|
|
TCPConnectBufferSize,
|
|
|
|
),
|
|
|
|
TLSHandshake: make(
|
|
|
|
chan *model.ArchivalTLSOrQUICHandshakeResult,
|
|
|
|
TLSHandshakeBufferSize,
|
|
|
|
),
|
|
|
|
TimeNowFn: nil, // use default
|
|
|
|
ZeroTime: zeroTime,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-11 18:18:25 +02:00
|
|
|
// newStdlibResolver indirectly calls the passed netxlite.NewStdlibResolver
|
|
|
|
// thus allowing us to mock this function for testing
|
|
|
|
func (tx *Trace) newStdlibResolver(logger model.Logger) model.Resolver {
|
|
|
|
if tx.NewStdlibResolverFn != nil {
|
|
|
|
return tx.NewStdlibResolverFn(logger)
|
|
|
|
}
|
|
|
|
return netxlite.NewStdlibResolver(logger)
|
|
|
|
}
|
|
|
|
|
2022-08-11 16:00:37 +02:00
|
|
|
// newParallelUDPResolver indirectly calls the passed netxlite.NewParallerUDPResolver
|
|
|
|
// thus allowing us to mock this function for testing
|
|
|
|
func (tx *Trace) newParallelUDPResolver(logger model.Logger, dialer model.Dialer, address string) model.Resolver {
|
|
|
|
if tx.NewParallelUDPResolverFn != nil {
|
|
|
|
return tx.NewParallelUDPResolverFn(logger, dialer, address)
|
|
|
|
}
|
|
|
|
return netxlite.NewParallelUDPResolver(logger, dialer, address)
|
|
|
|
}
|
|
|
|
|
|
|
|
// newParallelDNSOverHTTPSResolver indirectly calls the passed netxlite.NewParallerDNSOverHTTPSResolver
|
|
|
|
// thus allowing us to mock this function for testing
|
|
|
|
func (tx *Trace) newParallelDNSOverHTTPSResolver(logger model.Logger, URL string) model.Resolver {
|
|
|
|
if tx.NewParallelDNSOverHTTPSResolverFn != nil {
|
|
|
|
return tx.NewParallelDNSOverHTTPSResolverFn(logger, URL)
|
|
|
|
}
|
|
|
|
return netxlite.NewParallelDNSOverHTTPSResolver(logger, URL)
|
|
|
|
}
|
|
|
|
|
2022-07-01 12:22:22 +02:00
|
|
|
// newDialerWithoutResolver indirectly calls netxlite.NewDialerWithoutResolver
|
2022-07-08 19:42:24 +02:00
|
|
|
// thus allowing us to mock this func for testing.
|
2022-07-01 12:22:22 +02:00
|
|
|
func (tx *Trace) newDialerWithoutResolver(dl model.DebugLogger) model.Dialer {
|
|
|
|
if tx.NewDialerWithoutResolverFn != nil {
|
|
|
|
return tx.NewDialerWithoutResolverFn(dl)
|
|
|
|
}
|
|
|
|
return netxlite.NewDialerWithoutResolver(dl)
|
|
|
|
}
|
|
|
|
|
|
|
|
// newTLSHandshakerStdlib indirectly calls netxlite.NewTLSHandshakerStdlib
|
|
|
|
// thus allowing us to mock this func for testing.
|
|
|
|
func (tx *Trace) newTLSHandshakerStdlib(dl model.DebugLogger) model.TLSHandshaker {
|
|
|
|
if tx.NewTLSHandshakerStdlibFn != nil {
|
|
|
|
return tx.NewTLSHandshakerStdlibFn(dl)
|
|
|
|
}
|
|
|
|
return netxlite.NewTLSHandshakerStdlib(dl)
|
|
|
|
}
|
|
|
|
|
|
|
|
// TimeNow implements model.Trace.TimeNow.
|
|
|
|
func (tx *Trace) TimeNow() time.Time {
|
|
|
|
if tx.TimeNowFn != nil {
|
|
|
|
return tx.TimeNowFn()
|
|
|
|
}
|
|
|
|
return time.Now()
|
|
|
|
}
|
|
|
|
|
|
|
|
// TimeSince is equivalent to Trace.TimeNow().Sub(t0).
|
|
|
|
func (tx *Trace) TimeSince(t0 time.Time) time.Duration {
|
|
|
|
return tx.TimeNow().Sub(t0)
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ model.Trace = &Trace{}
|