2022-01-03 13:53:23 +01:00
|
|
|
package model
|
|
|
|
|
2022-05-25 17:03:58 +02:00
|
|
|
//
|
|
|
|
// Network extensions
|
|
|
|
//
|
|
|
|
|
2022-01-03 13:53:23 +01:00
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"crypto/tls"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
|
|
|
"syscall"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/lucas-clemente/quic-go"
|
|
|
|
)
|
|
|
|
|
2022-05-25 17:03:58 +02:00
|
|
|
// DNSResponse is a parsed DNS response ready for further processing.
|
|
|
|
type DNSResponse interface {
|
|
|
|
// Query is the query associated with this response.
|
|
|
|
Query() DNSQuery
|
2022-01-03 13:53:23 +01:00
|
|
|
|
2022-05-25 17:03:58 +02:00
|
|
|
// Bytes returns the bytes from which we parsed the query.
|
|
|
|
Bytes() []byte
|
2022-01-03 13:53:23 +01:00
|
|
|
|
2022-05-25 17:03:58 +02:00
|
|
|
// Rcode returns the response's Rcode.
|
|
|
|
Rcode() int
|
2022-05-14 19:38:46 +02:00
|
|
|
|
2022-05-25 17:03:58 +02:00
|
|
|
// DecodeHTTPS returns information gathered from all the HTTPS
|
|
|
|
// records found inside of this response.
|
|
|
|
DecodeHTTPS() (*HTTPSSvc, error)
|
2022-05-16 10:46:53 +02:00
|
|
|
|
2022-05-25 17:03:58 +02:00
|
|
|
// DecodeLookupHost returns the addresses in the response matching
|
|
|
|
// the original query type (one of A and AAAA).
|
|
|
|
DecodeLookupHost() ([]string, error)
|
|
|
|
|
|
|
|
// DecodeNS returns all the NS entries in this response.
|
|
|
|
DecodeNS() ([]*net.NS, error)
|
2022-08-23 13:04:00 +02:00
|
|
|
|
|
|
|
// DecodeCNAME returns the first CNAME entry in this response.
|
|
|
|
DecodeCNAME() (string, error)
|
2022-05-25 17:03:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// The DNSDecoder decodes DNS responses.
|
|
|
|
type DNSDecoder interface {
|
|
|
|
// DecodeResponse decodes a DNS response message.
|
2022-05-14 19:38:46 +02:00
|
|
|
//
|
|
|
|
// Arguments:
|
|
|
|
//
|
|
|
|
// - data is the raw reply
|
|
|
|
//
|
2022-05-16 11:17:30 +02:00
|
|
|
// This function fails if we cannot parse data as a DNS
|
2022-05-25 17:03:58 +02:00
|
|
|
// message or the message is not a response.
|
2022-05-14 19:38:46 +02:00
|
|
|
//
|
2022-05-25 17:03:58 +02:00
|
|
|
// Regarding the returned response, remember that the Rcode
|
|
|
|
// MAY still be nonzero (this method does not treat a nonzero
|
|
|
|
// Rcode as an error when parsing the response).
|
|
|
|
DecodeResponse(data []byte, query DNSQuery) (DNSResponse, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
// DNSQuery is an encoded DNS query ready to be sent using a DNSTransport.
|
|
|
|
type DNSQuery interface {
|
|
|
|
// Domain is the domain we're querying for.
|
|
|
|
Domain() string
|
|
|
|
|
|
|
|
// Type is the query type.
|
|
|
|
Type() uint16
|
|
|
|
|
|
|
|
// Bytes serializes the query to bytes. This function may fail if we're not
|
|
|
|
// able to correctly encode the domain into a query message.
|
2022-05-14 19:38:46 +02:00
|
|
|
//
|
2022-05-25 17:03:58 +02:00
|
|
|
// The value returned by this function WILL be memoized after the first call,
|
|
|
|
// so you SHOULD create a new DNSQuery if you need to retry a query.
|
|
|
|
Bytes() ([]byte, error)
|
|
|
|
|
|
|
|
// ID returns the query ID.
|
|
|
|
ID() uint16
|
2022-01-03 13:53:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// The DNSEncoder encodes DNS queries to bytes
|
|
|
|
type DNSEncoder interface {
|
|
|
|
// Encode transforms its arguments into a serialized DNS query.
|
|
|
|
//
|
2022-05-25 17:03:58 +02:00
|
|
|
// Every time you call Encode, you get a new DNSQuery value
|
|
|
|
// using a query ID selected at random.
|
|
|
|
//
|
|
|
|
// Serialization to bytes is lazy to acommodate DNS transports that
|
|
|
|
// do not need to serialize and send bytes, e.g., getaddrinfo.
|
|
|
|
//
|
|
|
|
// You serialize to bytes using DNSQuery.Bytes. This operation MAY fail
|
|
|
|
// if the domain name cannot be packed into a DNS message (e.g., it is
|
|
|
|
// too long to fit into the message).
|
|
|
|
//
|
2022-01-03 13:53:23 +01:00
|
|
|
// Arguments:
|
|
|
|
//
|
|
|
|
// - domain is the domain for the query (e.g., x.org);
|
|
|
|
//
|
|
|
|
// - qtype is the query type (e.g., dns.TypeA);
|
|
|
|
//
|
|
|
|
// - padding is whether to add padding to the query.
|
|
|
|
//
|
2022-05-25 17:03:58 +02:00
|
|
|
// This function will transform the domain into an FQDN is it's not
|
|
|
|
// already expressed in the FQDN format.
|
|
|
|
Encode(domain string, qtype uint16, padding bool) DNSQuery
|
2022-01-03 13:53:23 +01:00
|
|
|
}
|
|
|
|
|
2022-06-01 11:10:08 +02:00
|
|
|
// DNSTransportWrapper is a type that takes in input a DNSTransport
|
|
|
|
// and returns in output a wrapped DNSTransport.
|
|
|
|
type DNSTransportWrapper interface {
|
|
|
|
WrapDNSTransport(txp DNSTransport) DNSTransport
|
|
|
|
}
|
|
|
|
|
2022-01-03 13:53:23 +01:00
|
|
|
// DNSTransport represents an abstract DNS transport.
|
|
|
|
type DNSTransport interface {
|
|
|
|
// RoundTrip sends a DNS query and receives the reply.
|
2022-05-25 17:03:58 +02:00
|
|
|
RoundTrip(ctx context.Context, query DNSQuery) (DNSResponse, error)
|
2022-01-03 13:53:23 +01:00
|
|
|
|
|
|
|
// RequiresPadding returns whether this transport needs padding.
|
|
|
|
RequiresPadding() bool
|
|
|
|
|
|
|
|
// Network is the network of the round tripper (e.g. "dot").
|
|
|
|
Network() string
|
|
|
|
|
|
|
|
// Address is the address of the round tripper (e.g. "1.1.1.1:853").
|
|
|
|
Address() string
|
|
|
|
|
|
|
|
// CloseIdleConnections closes idle connections, if any.
|
|
|
|
CloseIdleConnections()
|
|
|
|
}
|
|
|
|
|
2022-06-01 08:31:20 +02:00
|
|
|
// DialerWrapper is a type that takes in input a Dialer
|
|
|
|
// and returns in output a wrapped Dialer.
|
|
|
|
type DialerWrapper interface {
|
|
|
|
WrapDialer(d Dialer) Dialer
|
|
|
|
}
|
|
|
|
|
2022-01-03 13:53:23 +01:00
|
|
|
// SimpleDialer establishes network connections.
|
|
|
|
type SimpleDialer interface {
|
|
|
|
// DialContext behaves like net.Dialer.DialContext.
|
|
|
|
DialContext(ctx context.Context, network, address string) (net.Conn, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Dialer is a SimpleDialer with the possibility of closing open connections.
|
|
|
|
type Dialer interface {
|
|
|
|
// A Dialer is also a SimpleDialer.
|
|
|
|
SimpleDialer
|
|
|
|
|
|
|
|
// CloseIdleConnections closes idle connections, if any.
|
|
|
|
CloseIdleConnections()
|
|
|
|
}
|
|
|
|
|
|
|
|
// HTTPClient is an http.Client-like interface.
|
|
|
|
type HTTPClient interface {
|
|
|
|
Do(req *http.Request) (*http.Response, error)
|
|
|
|
CloseIdleConnections()
|
|
|
|
}
|
|
|
|
|
|
|
|
// HTTPTransport is an http.Transport-like structure.
|
|
|
|
type HTTPTransport interface {
|
feature: merge measurex and netx archival layer (1/N) (#663)
This diff introduces a new package called `./internal/archival`. This package collects data from `./internal/model` network interfaces (e.g., `Dialer`, `QUICDialer`, `HTTPTransport`), saves such data into an internal tabular data format suitable for on-line processing and analysis, and allows exporting data into the OONI data format.
The code for collecting and the internal tabular data formats are adapted from `measurex`. The code for formatting and exporting OONI data-format-compliant structures is adapted from `netx/archival`.
My original objective was to _also_ (1) fully replace `netx/archival` with this package and (2) adapt `measurex` to use this package rather than its own code. Both operations seem easily feasible because: (a) this code is `measurex` code without extensions that are `measurex` related, which will need to be added back as part of the process; (b) the API provided by this code allows for trivially converting from using `netx/archival` to using this code.
Yet, both changes should not be taken lightly. After implementing them, there's need to spend some time doing QA and ensuring all nettests work as intended. However, I am planning a release in the next two weeks, and this QA task is likely going to defer the release. For this reason, I have chosen to commit the work done so far into the tree and defer the second part of this refactoring for a later moment in time. (This explains why the title mentions "1/N").
On a more high-level perspective, it would also be beneficial, I guess, to explain _why_ I am doing these changes. There are two intertwined reasons. The first reason is that `netx/archival` has shortcomings deriving from its original https://github.com/ooni/netx legacy. The most relevant shortcoming is that it saves all kind of data into the same tabular structure named `Event`. This design choice is unfortunate because it does not allow one to apply data-type specific logic when processing the results. In turn, this choice results in complex processing code. Therefore, I believe that replacing the code with event-specific data structures is clearly an improvement in terms of code maintainability and would quite likely lead us to more confidently change and evolve the codebase.
The second reason why I would like to move forward these changes is to unify the codepaths used for measuring. At this point in time, we basically have two codepaths: `./internal/engine/netx` and `./internal/measurex`. They both have pros and cons and I don't think we want to rewrite whole experiments using `netx`. Rather, what we probably want is to gradually merge these two codepaths such that `netx` is a set of abstractions on top of `measurex` (which is more low-level and has a more-easily-testable design). Because saving events and generating an archival data format out of them consists of at least 50% of the complexity of both `netx` and `measurex`, it seems reasonable to unify this archival-related part of the two codebases as the first step.
At the highest level of abstraction, these changes are part of the train of changes which will eventually lead us to bless `websteps` as a first class citizen in OONI land. Because `websteps` requires different underlying primitives, I chose to develop these primitives from scratch rather than wrestling with `netx`, which used another model. The model used by `websteps` is that we perform each operation in isolation and immediately we save the results, while `netx` creates whole data structures and collects all the events happening via tracing. We believe the model used by `websteps` to be better because it does not require your code to figure out everything that happened after the measurement, which is a source of subtle bugs in the current implementation. So, when I started implementing websteps I extracted the bits of `netx` that could also be beneficial to `websteps` into a separate library, thus `netxlite` was born.
The reference issue describing merging the archival of `netx` and `measurex` is https://github.com/ooni/probe/issues/1957. As of this writing the issue still references the original plan, which I could not complete by the end of this Sprint, so I am going to adapt the text of the issue to only refer to what was done in here next. Of course, I also need follow-up issues.
2022-01-14 12:13:10 +01:00
|
|
|
// Network returns the network used by the transport, which
|
2022-09-08 17:19:59 +02:00
|
|
|
// should be one of "tcp" and "udp".
|
feature: merge measurex and netx archival layer (1/N) (#663)
This diff introduces a new package called `./internal/archival`. This package collects data from `./internal/model` network interfaces (e.g., `Dialer`, `QUICDialer`, `HTTPTransport`), saves such data into an internal tabular data format suitable for on-line processing and analysis, and allows exporting data into the OONI data format.
The code for collecting and the internal tabular data formats are adapted from `measurex`. The code for formatting and exporting OONI data-format-compliant structures is adapted from `netx/archival`.
My original objective was to _also_ (1) fully replace `netx/archival` with this package and (2) adapt `measurex` to use this package rather than its own code. Both operations seem easily feasible because: (a) this code is `measurex` code without extensions that are `measurex` related, which will need to be added back as part of the process; (b) the API provided by this code allows for trivially converting from using `netx/archival` to using this code.
Yet, both changes should not be taken lightly. After implementing them, there's need to spend some time doing QA and ensuring all nettests work as intended. However, I am planning a release in the next two weeks, and this QA task is likely going to defer the release. For this reason, I have chosen to commit the work done so far into the tree and defer the second part of this refactoring for a later moment in time. (This explains why the title mentions "1/N").
On a more high-level perspective, it would also be beneficial, I guess, to explain _why_ I am doing these changes. There are two intertwined reasons. The first reason is that `netx/archival` has shortcomings deriving from its original https://github.com/ooni/netx legacy. The most relevant shortcoming is that it saves all kind of data into the same tabular structure named `Event`. This design choice is unfortunate because it does not allow one to apply data-type specific logic when processing the results. In turn, this choice results in complex processing code. Therefore, I believe that replacing the code with event-specific data structures is clearly an improvement in terms of code maintainability and would quite likely lead us to more confidently change and evolve the codebase.
The second reason why I would like to move forward these changes is to unify the codepaths used for measuring. At this point in time, we basically have two codepaths: `./internal/engine/netx` and `./internal/measurex`. They both have pros and cons and I don't think we want to rewrite whole experiments using `netx`. Rather, what we probably want is to gradually merge these two codepaths such that `netx` is a set of abstractions on top of `measurex` (which is more low-level and has a more-easily-testable design). Because saving events and generating an archival data format out of them consists of at least 50% of the complexity of both `netx` and `measurex`, it seems reasonable to unify this archival-related part of the two codebases as the first step.
At the highest level of abstraction, these changes are part of the train of changes which will eventually lead us to bless `websteps` as a first class citizen in OONI land. Because `websteps` requires different underlying primitives, I chose to develop these primitives from scratch rather than wrestling with `netx`, which used another model. The model used by `websteps` is that we perform each operation in isolation and immediately we save the results, while `netx` creates whole data structures and collects all the events happening via tracing. We believe the model used by `websteps` to be better because it does not require your code to figure out everything that happened after the measurement, which is a source of subtle bugs in the current implementation. So, when I started implementing websteps I extracted the bits of `netx` that could also be beneficial to `websteps` into a separate library, thus `netxlite` was born.
The reference issue describing merging the archival of `netx` and `measurex` is https://github.com/ooni/probe/issues/1957. As of this writing the issue still references the original plan, which I could not complete by the end of this Sprint, so I am going to adapt the text of the issue to only refer to what was done in here next. Of course, I also need follow-up issues.
2022-01-14 12:13:10 +01:00
|
|
|
Network() string
|
|
|
|
|
2022-01-03 13:53:23 +01:00
|
|
|
// RoundTrip performs the HTTP round trip.
|
|
|
|
RoundTrip(req *http.Request) (*http.Response, error)
|
|
|
|
|
|
|
|
// CloseIdleConnections closes idle connections.
|
|
|
|
CloseIdleConnections()
|
|
|
|
}
|
|
|
|
|
|
|
|
// HTTPSSvc is the reply to an HTTPS DNS query.
|
|
|
|
type HTTPSSvc struct {
|
|
|
|
// ALPN contains the ALPNs inside the HTTPS reply.
|
|
|
|
ALPN []string
|
|
|
|
|
|
|
|
// IPv4 contains the IPv4 hints (which may be empty).
|
|
|
|
IPv4 []string
|
|
|
|
|
|
|
|
// IPv6 contains the IPv6 hints (which may be empty).
|
|
|
|
IPv6 []string
|
|
|
|
}
|
|
|
|
|
|
|
|
// QUICListener listens for QUIC connections.
|
|
|
|
type QUICListener interface {
|
|
|
|
// Listen creates a new listening UDPLikeConn.
|
|
|
|
Listen(addr *net.UDPAddr) (UDPLikeConn, error)
|
|
|
|
}
|
|
|
|
|
2022-06-01 08:31:20 +02:00
|
|
|
// QUICDialerWrapper is a type that takes in input a QUICDialer
|
|
|
|
// and returns in output a wrapped QUICDialer.
|
|
|
|
type QUICDialerWrapper interface {
|
|
|
|
WrapQUICDialer(qd QUICDialer) QUICDialer
|
|
|
|
}
|
|
|
|
|
2022-01-03 13:53:23 +01:00
|
|
|
// QUICDialer dials QUIC sessions.
|
|
|
|
type QUICDialer interface {
|
|
|
|
// DialContext establishes a new QUIC session using the given
|
|
|
|
// network and address. The tlsConfig and the quicConfig arguments
|
|
|
|
// MUST NOT be nil. Returns either the session or an error.
|
|
|
|
//
|
|
|
|
// Recommended tlsConfig setup:
|
|
|
|
//
|
|
|
|
// - set ServerName to be the SNI;
|
|
|
|
//
|
|
|
|
// - set RootCAs to NewDefaultCertPool();
|
|
|
|
//
|
|
|
|
// - set NextProtos to []string{"h3"}.
|
|
|
|
//
|
|
|
|
// Typically, you want to pass `&quic.Config{}` as quicConfig.
|
2022-08-19 11:26:50 +02:00
|
|
|
DialContext(ctx context.Context, address string,
|
2022-05-06 12:24:03 +02:00
|
|
|
tlsConfig *tls.Config, quicConfig *quic.Config) (quic.EarlyConnection, error)
|
2022-01-03 13:53:23 +01:00
|
|
|
|
|
|
|
// CloseIdleConnections closes idle connections, if any.
|
|
|
|
CloseIdleConnections()
|
|
|
|
}
|
|
|
|
|
2022-08-23 11:43:44 +02:00
|
|
|
// Resolver performs domain name resolutions.
|
|
|
|
type Resolver interface {
|
2022-01-03 13:53:23 +01:00
|
|
|
// LookupHost behaves like net.Resolver.LookupHost.
|
|
|
|
LookupHost(ctx context.Context, hostname string) (addrs []string, err error)
|
|
|
|
|
getaddrinfo: fix CGO_ENABLED=0 and record resolver type (#765)
After https://github.com/ooni/probe-cli/pull/764, the build for
CGO_ENABLED=0 has been broken for miniooni:
https://github.com/ooni/probe-cli/runs/6636995859?check_suite_focus=true
Likewise, it's not possible to run tests with CGO_ENABLED=0.
To make tests work with `CGO_ENABLED=0`, I needed to sacrifice some
unit tests run for the CGO case. It is not fully clear to me what was happening
here, but basically `getaddrinfo_cgo_test.go` was compiled with CGO
being disabled, even though the ``//go:build cgo` flag was specified.
Additionally, @hellais previously raised a valid point in the review
of https://github.com/ooni/probe-cli/pull/698:
> Another issue we should consider is that, if I understand how
> this works correctly, depending on whether or not we have built
> with CGO_ENABLED=0 on or not, we are going to be measuring
> things in a different way (using our cgo inspired getaddrinfo
> implementation or using netgo). This might present issues when
> analyzing or interpreting the data.
>
> Do we perhaps want to add some field to the output data format that
> gives us an indication of which DNS resolution code was used to
> generate the the metric?
This comment is relevant to the current commit because
https://github.com/ooni/probe-cli/pull/698 is the previous
iteration of https://github.com/ooni/probe-cli/pull/764.
So, while fixing the build and test issues, let us also distinguish
between the CGO_ENABLED=1 and CGO_ENABLED=0 cases.
Before this commit, OONI used "system" to indicate the case where
we were using net.DefaultResolver. This behavior dates back to the
Measurement Kit days. While it is true that ooni/probe-engine and
ooni/probe-cli could have been using netgo in the past when we
said "system" as the resolver, it also seems reasonable to continue
to use "system" top indicate getaddrinfo.
So, the choice here is basically to use "netgo" from now on to
indicate the cases in which we were built with CGO_ENABLED=0.
This change will need to be documented into ooni/spec along with
the introduction of the `android_dns_cache_no_data` error.
## Checklist
- [x] I have read the [contribution guidelines](https://github.com/ooni/probe-cli/blob/master/CONTRIBUTING.md)
- [x] reference issue for this pull request: https://github.com/ooni/probe/issues/2029
- [x] if you changed anything related how experiments work and you need to reflect these changes in the ooni/spec repository, please link to the related ooni/spec pull request: https://github.com/ooni/spec/pull/242
2022-05-30 07:34:25 +02:00
|
|
|
// Network returns the resolver type. It should be one of:
|
|
|
|
//
|
2022-05-30 10:06:53 +02:00
|
|
|
// - go: means we're using whatever resolver the Go stdlib uses
|
|
|
|
// depending on the current build configuration;
|
getaddrinfo: fix CGO_ENABLED=0 and record resolver type (#765)
After https://github.com/ooni/probe-cli/pull/764, the build for
CGO_ENABLED=0 has been broken for miniooni:
https://github.com/ooni/probe-cli/runs/6636995859?check_suite_focus=true
Likewise, it's not possible to run tests with CGO_ENABLED=0.
To make tests work with `CGO_ENABLED=0`, I needed to sacrifice some
unit tests run for the CGO case. It is not fully clear to me what was happening
here, but basically `getaddrinfo_cgo_test.go` was compiled with CGO
being disabled, even though the ``//go:build cgo` flag was specified.
Additionally, @hellais previously raised a valid point in the review
of https://github.com/ooni/probe-cli/pull/698:
> Another issue we should consider is that, if I understand how
> this works correctly, depending on whether or not we have built
> with CGO_ENABLED=0 on or not, we are going to be measuring
> things in a different way (using our cgo inspired getaddrinfo
> implementation or using netgo). This might present issues when
> analyzing or interpreting the data.
>
> Do we perhaps want to add some field to the output data format that
> gives us an indication of which DNS resolution code was used to
> generate the the metric?
This comment is relevant to the current commit because
https://github.com/ooni/probe-cli/pull/698 is the previous
iteration of https://github.com/ooni/probe-cli/pull/764.
So, while fixing the build and test issues, let us also distinguish
between the CGO_ENABLED=1 and CGO_ENABLED=0 cases.
Before this commit, OONI used "system" to indicate the case where
we were using net.DefaultResolver. This behavior dates back to the
Measurement Kit days. While it is true that ooni/probe-engine and
ooni/probe-cli could have been using netgo in the past when we
said "system" as the resolver, it also seems reasonable to continue
to use "system" top indicate getaddrinfo.
So, the choice here is basically to use "netgo" from now on to
indicate the cases in which we were built with CGO_ENABLED=0.
This change will need to be documented into ooni/spec along with
the introduction of the `android_dns_cache_no_data` error.
## Checklist
- [x] I have read the [contribution guidelines](https://github.com/ooni/probe-cli/blob/master/CONTRIBUTING.md)
- [x] reference issue for this pull request: https://github.com/ooni/probe/issues/2029
- [x] if you changed anything related how experiments work and you need to reflect these changes in the ooni/spec repository, please link to the related ooni/spec pull request: https://github.com/ooni/spec/pull/242
2022-05-30 07:34:25 +02:00
|
|
|
//
|
2022-05-30 10:06:53 +02:00
|
|
|
// - system: means we've been compiled with `CGO_ENABLED=1`
|
|
|
|
// so we can bypass the go resolver and call getaddrinfo directly;
|
getaddrinfo: fix CGO_ENABLED=0 and record resolver type (#765)
After https://github.com/ooni/probe-cli/pull/764, the build for
CGO_ENABLED=0 has been broken for miniooni:
https://github.com/ooni/probe-cli/runs/6636995859?check_suite_focus=true
Likewise, it's not possible to run tests with CGO_ENABLED=0.
To make tests work with `CGO_ENABLED=0`, I needed to sacrifice some
unit tests run for the CGO case. It is not fully clear to me what was happening
here, but basically `getaddrinfo_cgo_test.go` was compiled with CGO
being disabled, even though the ``//go:build cgo` flag was specified.
Additionally, @hellais previously raised a valid point in the review
of https://github.com/ooni/probe-cli/pull/698:
> Another issue we should consider is that, if I understand how
> this works correctly, depending on whether or not we have built
> with CGO_ENABLED=0 on or not, we are going to be measuring
> things in a different way (using our cgo inspired getaddrinfo
> implementation or using netgo). This might present issues when
> analyzing or interpreting the data.
>
> Do we perhaps want to add some field to the output data format that
> gives us an indication of which DNS resolution code was used to
> generate the the metric?
This comment is relevant to the current commit because
https://github.com/ooni/probe-cli/pull/698 is the previous
iteration of https://github.com/ooni/probe-cli/pull/764.
So, while fixing the build and test issues, let us also distinguish
between the CGO_ENABLED=1 and CGO_ENABLED=0 cases.
Before this commit, OONI used "system" to indicate the case where
we were using net.DefaultResolver. This behavior dates back to the
Measurement Kit days. While it is true that ooni/probe-engine and
ooni/probe-cli could have been using netgo in the past when we
said "system" as the resolver, it also seems reasonable to continue
to use "system" top indicate getaddrinfo.
So, the choice here is basically to use "netgo" from now on to
indicate the cases in which we were built with CGO_ENABLED=0.
This change will need to be documented into ooni/spec along with
the introduction of the `android_dns_cache_no_data` error.
## Checklist
- [x] I have read the [contribution guidelines](https://github.com/ooni/probe-cli/blob/master/CONTRIBUTING.md)
- [x] reference issue for this pull request: https://github.com/ooni/probe/issues/2029
- [x] if you changed anything related how experiments work and you need to reflect these changes in the ooni/spec repository, please link to the related ooni/spec pull request: https://github.com/ooni/spec/pull/242
2022-05-30 07:34:25 +02:00
|
|
|
//
|
|
|
|
// - udp: is a custom DNS-over-UDP resolver;
|
|
|
|
//
|
|
|
|
// - tcp: is a custom DNS-over-TCP resolver;
|
|
|
|
//
|
|
|
|
// - dot: is a custom DNS-over-TLS resolver;
|
|
|
|
//
|
|
|
|
// - doh: is a custom DNS-over-HTTPS resolver;
|
|
|
|
//
|
|
|
|
// - doh3: is a custom DNS-over-HTTP3 resolver.
|
2022-05-30 10:06:53 +02:00
|
|
|
//
|
|
|
|
// See https://github.com/ooni/probe/issues/2029#issuecomment-1140805266
|
|
|
|
// for an explanation of why it would not be proper to call "netgo" the
|
|
|
|
// resolver we get by default from the standard library.
|
2022-01-03 13:53:23 +01:00
|
|
|
Network() string
|
|
|
|
|
|
|
|
// Address returns the resolver address (e.g., 8.8.8.8:53).
|
|
|
|
Address() string
|
|
|
|
|
|
|
|
// CloseIdleConnections closes idle connections, if any.
|
|
|
|
CloseIdleConnections()
|
|
|
|
|
|
|
|
// LookupHTTPS issues an HTTPS query for a domain.
|
|
|
|
LookupHTTPS(
|
|
|
|
ctx context.Context, domain string) (*HTTPSSvc, error)
|
2022-05-16 10:46:53 +02:00
|
|
|
|
|
|
|
// LookupNS issues a NS query for a domain.
|
|
|
|
LookupNS(ctx context.Context, domain string) ([]*net.NS, error)
|
2022-01-03 13:53:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// TLSDialer is a Dialer dialing TLS connections.
|
|
|
|
type TLSDialer interface {
|
|
|
|
// CloseIdleConnections closes idle connections, if any.
|
|
|
|
CloseIdleConnections()
|
|
|
|
|
|
|
|
// DialTLSContext dials a TLS connection. This method will always return
|
|
|
|
// to you a oohttp.TLSConn, so you can always safely cast to it.
|
|
|
|
DialTLSContext(ctx context.Context, network, address string) (net.Conn, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
// TLSHandshaker is the generic TLS handshaker.
|
|
|
|
type TLSHandshaker interface {
|
|
|
|
// Handshake creates a new TLS connection from the given connection and
|
|
|
|
// the given config. This function DOES NOT take ownership of the connection
|
|
|
|
// and it's your responsibility to close it on failure.
|
|
|
|
//
|
|
|
|
// Recommended tlsConfig setup:
|
|
|
|
//
|
|
|
|
// - set ServerName to be the SNI;
|
|
|
|
//
|
|
|
|
// - set RootCAs to NewDefaultCertPool();
|
|
|
|
//
|
|
|
|
// - set NextProtos to []string{"h2", "http/1.1"} for HTTPS
|
|
|
|
// and []string{"dot"} for DNS-over-TLS.
|
|
|
|
//
|
|
|
|
// QUIRK: The returned connection will always implement the TLSConn interface
|
|
|
|
// exposed by ooni/oohttp. A future version of this interface may instead
|
|
|
|
// return directly a TLSConn to avoid unconditional castings.
|
|
|
|
Handshake(ctx context.Context, conn net.Conn, tlsConfig *tls.Config) (
|
|
|
|
net.Conn, tls.ConnectionState, error)
|
|
|
|
}
|
|
|
|
|
2022-07-01 12:22:22 +02:00
|
|
|
// Trace allows to collect measurement traces. A trace is injected into
|
|
|
|
// netx operations using context.WithValue. Netx code retrieves the trace
|
|
|
|
// using context.Value. See docs/design/dd-003-step-by-step.md for the
|
|
|
|
// design document explaining why we implemented context-based tracing.
|
|
|
|
type Trace interface {
|
|
|
|
// TimeNow returns the current time. Normally, this should be the same
|
|
|
|
// value returned by time.Now but you may want to manipulate the time
|
|
|
|
// returned when testing to have deterministic tests. To this end, you
|
|
|
|
// can use functionality exported by the ./internal/testingx pkg.
|
|
|
|
TimeNow() time.Time
|
|
|
|
|
2022-08-17 20:58:06 +02:00
|
|
|
// MaybeWrapNetConn possibly wraps a net.Conn with the caller trace. If there's no
|
|
|
|
// desire to wrap the net.Conn, this function just returns the original net.Conn.
|
|
|
|
//
|
|
|
|
// Arguments:
|
|
|
|
//
|
|
|
|
// - conn is the non-nil underlying net.Conn to be wrapped
|
|
|
|
MaybeWrapNetConn(conn net.Conn) net.Conn
|
|
|
|
|
|
|
|
// MaybeWrapUDPLikeConn is like MaybeWrapNetConn but for UDPLikeConn.
|
|
|
|
//
|
|
|
|
// Arguments:
|
|
|
|
//
|
|
|
|
// - conn is the non-nil underlying UDPLikeConn to be wrapped
|
|
|
|
MaybeWrapUDPLikeConn(conn UDPLikeConn) UDPLikeConn
|
|
|
|
|
2022-07-08 19:42:24 +02:00
|
|
|
// OnDNSRoundTripForLookupHost is used with a DNSTransport and called
|
|
|
|
// when the RoundTrip terminates.
|
|
|
|
//
|
|
|
|
// Arguments:
|
|
|
|
//
|
|
|
|
// - started is when we called transport.RoundTrip
|
|
|
|
//
|
|
|
|
// - reso is the parent resolver for the trace;
|
|
|
|
//
|
|
|
|
// - query is the non-nil DNS query we use for the RoundTrip
|
|
|
|
//
|
|
|
|
// - response is a valid DNS response, obtained after the RoundTrip;
|
|
|
|
//
|
|
|
|
// - addrs is the list of addresses obtained after the RoundTrip, which
|
|
|
|
// is empty if the RoundTrip failed
|
|
|
|
//
|
|
|
|
// - err is the result of DNSLookup; either an error or nil
|
|
|
|
//
|
|
|
|
// - finished is the time right after the RoundTrip
|
|
|
|
OnDNSRoundTripForLookupHost(started time.Time, reso Resolver, query DNSQuery,
|
|
|
|
response DNSResponse, addrs []string, err error, finished time.Time)
|
|
|
|
|
2022-08-22 14:21:32 +02:00
|
|
|
// OnDelayedDNSResponse is used with a DNSOverUDPTransport and called
|
|
|
|
// when we get delayed, unexpected DNS responses.
|
|
|
|
//
|
|
|
|
// Arguments:
|
|
|
|
//
|
|
|
|
// - started is when we started reading the delayed response;
|
|
|
|
//
|
|
|
|
// - txp is the DNS transport used with the resolver;
|
|
|
|
//
|
|
|
|
// - query is the non-nil DNS query we use for the RoundTrip;
|
|
|
|
//
|
|
|
|
// - response is the non-nil valid DNS response, obtained after some delay;
|
|
|
|
//
|
|
|
|
// - addrs is the list of addresses obtained after decoding the delayed response,
|
|
|
|
// which is empty if the response did not contain any addresses, which we
|
|
|
|
// extract by calling the DecodeLookupHost method.
|
|
|
|
//
|
|
|
|
// - err is the result of DecodeLookupHost: either an error or nil;
|
|
|
|
//
|
|
|
|
// - finished is when we have read the delayed response.
|
|
|
|
OnDelayedDNSResponse(started time.Time, txp DNSTransport, query DNSQuery,
|
|
|
|
resp DNSResponse, addrs []string, err error, finsihed time.Time) error
|
|
|
|
|
2022-07-01 12:22:22 +02:00
|
|
|
// OnConnectDone is called when connect terminates.
|
|
|
|
//
|
|
|
|
// Arguments:
|
|
|
|
//
|
|
|
|
// - started is when we called connect;
|
|
|
|
//
|
|
|
|
// - network is the network we're using (one of "tcp" and "udp");
|
|
|
|
//
|
|
|
|
// - domain is the domain for which we're calling connect. If the user called
|
|
|
|
// connect for an IP address and a port, then domain will be an IP address;
|
|
|
|
//
|
|
|
|
// - remoteAddr is the TCP endpoint with which we are connecting: it will
|
|
|
|
// consist of an IP address and a port (e.g., 8.8.8.8:443, [::1]:5421);
|
|
|
|
//
|
|
|
|
// - err is the result of connect: either an error or nil;
|
|
|
|
//
|
|
|
|
// - finished is when connect returned.
|
|
|
|
//
|
|
|
|
// The error passed to this function will always be wrapped such that the
|
|
|
|
// string returned by Error is an OONI error.
|
|
|
|
OnConnectDone(
|
|
|
|
started time.Time, network, domain, remoteAddr string, err error, finished time.Time)
|
|
|
|
|
|
|
|
// OnTLSHandshakeStart is called when the TLS handshake starts.
|
|
|
|
//
|
|
|
|
// Arguments:
|
|
|
|
//
|
|
|
|
// - now is the moment before we start the handshake;
|
|
|
|
//
|
|
|
|
// - remoteAddr is the TCP endpoint with which we are connecting: it will
|
|
|
|
// consist of an IP address and a port (e.g., 8.8.8.8:443, [::1]:5421);
|
|
|
|
//
|
|
|
|
// - config is the non-nil TLS config we're using.
|
|
|
|
OnTLSHandshakeStart(now time.Time, remoteAddr string, config *tls.Config)
|
|
|
|
|
|
|
|
// OnTLSHandshakeDone is called when the TLS handshake terminates.
|
|
|
|
//
|
|
|
|
// Arguments:
|
|
|
|
//
|
|
|
|
// - started is when we started the handshake;
|
|
|
|
//
|
|
|
|
// - remoteAddr is the TCP endpoint with which we are connecting: it will
|
|
|
|
// consist of an IP address and a port (e.g., 8.8.8.8:443, [::1]:5421);
|
|
|
|
//
|
|
|
|
// - config is the non-nil TLS config we're using;
|
|
|
|
//
|
|
|
|
// - state is the state of the TLS connection after the handshake, where all
|
|
|
|
// fields are zero-initialized if the handshake failed;
|
|
|
|
//
|
|
|
|
// - err is the result of the handshake: either an error or nil;
|
|
|
|
//
|
|
|
|
// - finished is right after the handshake.
|
|
|
|
//
|
|
|
|
// The error passed to this function will always be wrapped such that the
|
|
|
|
// string returned by Error is an OONI error.
|
|
|
|
OnTLSHandshakeDone(started time.Time, remoteAddr string, config *tls.Config,
|
|
|
|
state tls.ConnectionState, err error, finished time.Time)
|
2022-08-17 09:19:11 +02:00
|
|
|
|
|
|
|
// OnQUICHandshakeStart is called before the QUIC handshake.
|
|
|
|
//
|
|
|
|
// Arguments:
|
|
|
|
//
|
|
|
|
// - now is the moment before we start the handshake;
|
|
|
|
//
|
|
|
|
// - remoteAddr is the QUIC endpoint with which we are connecting: it will
|
|
|
|
// consist of an IP address and a port (e.g., 8.8.8.8:443, [::1]:5421);
|
|
|
|
//
|
|
|
|
// - config is the possibly-nil QUIC config we're using.
|
|
|
|
OnQUICHandshakeStart(now time.Time, remoteAddr string, quicConfig *quic.Config)
|
|
|
|
|
|
|
|
// OnQUICHandshakeDone is called after the QUIC handshake.
|
|
|
|
//
|
|
|
|
// Arguments:
|
|
|
|
//
|
|
|
|
// - started is when we started the handshake;
|
|
|
|
//
|
|
|
|
// - remoteAddr is the QUIC endpoint with which we are connecting: it will
|
|
|
|
// consist of an IP address and a port (e.g., 8.8.8.8:443, [::1]:5421);
|
|
|
|
//
|
|
|
|
// - qconn is the QUIC connection we receive after the handshake: either
|
|
|
|
// a valid quic.EarlyConnection or nil;
|
|
|
|
//
|
|
|
|
// - config is the non-nil TLS config we are using;
|
|
|
|
//
|
|
|
|
// - err is the result of the handshake: either an error or nil;
|
|
|
|
//
|
|
|
|
// - finished is right after the handshake.
|
|
|
|
//
|
|
|
|
// The error passed to this function will always be wrapped such that the
|
|
|
|
// string returned by Error is an OONI error.
|
|
|
|
OnQUICHandshakeDone(started time.Time, remoteAddr string, qconn quic.EarlyConnection,
|
|
|
|
config *tls.Config, err error, finished time.Time)
|
2022-07-01 12:22:22 +02:00
|
|
|
}
|
|
|
|
|
2022-01-03 13:53:23 +01:00
|
|
|
// UDPLikeConn is a net.PacketConn with some extra functions
|
|
|
|
// required to convince the QUIC library (lucas-clemente/quic-go)
|
|
|
|
// to inflate the receive buffer of the connection.
|
|
|
|
//
|
|
|
|
// The QUIC library will treat this connection as a "dumb"
|
|
|
|
// net.PacketConn, calling its ReadFrom and WriteTo methods
|
|
|
|
// as opposed to more efficient methods that are available
|
|
|
|
// under Linux and (maybe?) FreeBSD.
|
|
|
|
//
|
|
|
|
// It seems fine to avoid performance optimizations, because
|
|
|
|
// they would complicate the implementation on our side and
|
|
|
|
// our use cases (blocking and heavy throttling) do not seem
|
|
|
|
// to require such optimizations.
|
|
|
|
//
|
|
|
|
// See https://github.com/ooni/probe/issues/1754 for a more
|
|
|
|
// comprehensive discussion of UDPLikeConn.
|
|
|
|
type UDPLikeConn interface {
|
|
|
|
// An UDPLikeConn is a net.PacketConn conn.
|
|
|
|
net.PacketConn
|
|
|
|
|
|
|
|
// SetReadBuffer allows setting the read buffer.
|
|
|
|
SetReadBuffer(bytes int) error
|
|
|
|
|
|
|
|
// SyscallConn returns a conn suitable for calling syscalls,
|
|
|
|
// which is also instrumental to setting the read buffer.
|
|
|
|
SyscallConn() (syscall.RawConn, error)
|
|
|
|
}
|