2021-09-28 12:42:01 +02:00
|
|
|
package netxlite
|
2021-09-07 17:09:30 +02:00
|
|
|
|
2021-11-06 17:49:58 +01:00
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
)
|
2021-09-08 17:42:36 +02:00
|
|
|
|
2021-09-07 17:09:30 +02:00
|
|
|
// ErrWrapper is our error wrapper for Go errors. The key objective of
|
|
|
|
// this structure is to properly set Failure, which is also returned by
|
2021-09-07 20:39:32 +02:00
|
|
|
// the Error() method, to be one of the OONI failure strings.
|
|
|
|
//
|
|
|
|
// OONI failure strings are defined in the github.com/ooni/spec repo
|
|
|
|
// at https://github.com/ooni/spec/blob/master/data-formats/df-007-errors.md.
|
2021-09-07 17:09:30 +02:00
|
|
|
type ErrWrapper struct {
|
|
|
|
// Failure is the OONI failure string. The failure strings are
|
|
|
|
// loosely backward compatible with Measurement Kit.
|
|
|
|
//
|
|
|
|
// This is either one of the FailureXXX strings or any other
|
2021-09-07 20:39:32 +02:00
|
|
|
// string like `unknown_failure: ...`. The latter represents an
|
2021-09-07 17:09:30 +02:00
|
|
|
// error that we have not yet mapped to a failure.
|
|
|
|
Failure string
|
|
|
|
|
2021-09-07 20:39:32 +02:00
|
|
|
// Operation is the operation that failed.
|
|
|
|
//
|
2022-01-07 17:31:21 +01:00
|
|
|
// If possible, the Operation string SHOULD be a _major_
|
|
|
|
// operation. Major operations are:
|
2021-09-07 17:09:30 +02:00
|
|
|
//
|
|
|
|
// - ResolveOperation: resolving a domain name failed
|
|
|
|
// - ConnectOperation: connecting to an IP failed
|
|
|
|
// - TLSHandshakeOperation: TLS handshaking failed
|
2022-01-07 17:31:21 +01:00
|
|
|
// - QUICHandshakeOperation: QUIC handshaking failed
|
2021-09-07 17:09:30 +02:00
|
|
|
// - HTTPRoundTripOperation: other errors during round trip
|
|
|
|
//
|
|
|
|
// Because a network connection doesn't necessarily know
|
|
|
|
// what is the current major operation we also have the
|
|
|
|
// following _minor_ operations:
|
|
|
|
//
|
|
|
|
// - CloseOperation: CLOSE failed
|
|
|
|
// - ReadOperation: READ failed
|
|
|
|
// - WriteOperation: WRITE failed
|
|
|
|
//
|
|
|
|
// If an ErrWrapper referring to a major operation is wrapping
|
|
|
|
// another ErrWrapper and such ErrWrapper already refers to
|
|
|
|
// a major operation, then the new ErrWrapper should use the
|
|
|
|
// child ErrWrapper major operation. Otherwise, it should use
|
|
|
|
// its own major operation. This way, the topmost wrapper is
|
|
|
|
// supposed to refer to the major operation that failed.
|
|
|
|
Operation string
|
|
|
|
|
|
|
|
// WrappedErr is the error that we're wrapping.
|
|
|
|
WrappedErr error
|
|
|
|
}
|
|
|
|
|
2021-09-29 20:21:25 +02:00
|
|
|
// Error returns the OONI failure string for this error.
|
2021-09-07 17:09:30 +02:00
|
|
|
func (e *ErrWrapper) Error() string {
|
|
|
|
return e.Failure
|
|
|
|
}
|
|
|
|
|
2021-09-29 20:21:25 +02:00
|
|
|
// Unwrap allows to access the underlying error.
|
2021-09-07 17:09:30 +02:00
|
|
|
func (e *ErrWrapper) Unwrap() error {
|
|
|
|
return e.WrappedErr
|
|
|
|
}
|
2021-09-08 17:42:36 +02:00
|
|
|
|
|
|
|
// MarshalJSON converts an ErrWrapper to a JSON value.
|
|
|
|
func (e *ErrWrapper) MarshalJSON() ([]byte, error) {
|
|
|
|
return json.Marshal(e.Failure)
|
|
|
|
}
|
2021-09-08 21:19:51 +02:00
|
|
|
|
2021-09-29 20:21:25 +02:00
|
|
|
// Classifier is the type of the function that maps a Go error
|
|
|
|
// to a OONI failure string defined at
|
|
|
|
// https://github.com/ooni/spec/blob/master/data-formats/df-007-errors.md.
|
2021-09-08 21:19:51 +02:00
|
|
|
type Classifier func(err error) string
|
|
|
|
|
|
|
|
// NewErrWrapper creates a new ErrWrapper using the given
|
|
|
|
// classifier, operation name, and underlying error.
|
|
|
|
//
|
|
|
|
// This function panics if classifier is nil, or operation
|
|
|
|
// is the empty string or error is nil.
|
2021-11-06 17:49:58 +01:00
|
|
|
//
|
|
|
|
// If the err argument has already been classified, the returned
|
|
|
|
// error wrapper will use the same classification string and
|
2022-01-07 17:31:21 +01:00
|
|
|
// will determine whether to keep the major operation as documented
|
|
|
|
// in the ErrWrapper.Operation documentation.
|
2021-09-08 21:19:51 +02:00
|
|
|
func NewErrWrapper(c Classifier, op string, err error) *ErrWrapper {
|
2021-11-06 17:49:58 +01:00
|
|
|
var wrapper *ErrWrapper
|
|
|
|
if errors.As(err, &wrapper) {
|
|
|
|
return &ErrWrapper{
|
|
|
|
Failure: wrapper.Failure,
|
2022-01-07 17:31:21 +01:00
|
|
|
Operation: classifyOperation(wrapper, op),
|
2021-11-06 17:49:58 +01:00
|
|
|
WrappedErr: err,
|
|
|
|
}
|
|
|
|
}
|
2021-09-08 21:19:51 +02:00
|
|
|
if c == nil {
|
|
|
|
panic("nil classifier")
|
|
|
|
}
|
|
|
|
if op == "" {
|
|
|
|
panic("empty op")
|
|
|
|
}
|
|
|
|
if err == nil {
|
|
|
|
panic("nil err")
|
|
|
|
}
|
|
|
|
return &ErrWrapper{
|
|
|
|
Failure: c(err),
|
|
|
|
Operation: op,
|
|
|
|
WrappedErr: err,
|
|
|
|
}
|
|
|
|
}
|
2021-09-27 14:55:47 +02:00
|
|
|
|
|
|
|
// NewTopLevelGenericErrWrapper wraps an error occurring at top
|
2021-09-29 20:21:25 +02:00
|
|
|
// level using ClassifyGenericError as classifier.
|
2021-11-06 17:49:58 +01:00
|
|
|
//
|
|
|
|
// If the err argument has already been classified, the returned
|
|
|
|
// error wrapper will use the same classification string and
|
|
|
|
// failed operation of the original error.
|
2021-09-27 14:55:47 +02:00
|
|
|
func NewTopLevelGenericErrWrapper(err error) *ErrWrapper {
|
2022-01-07 17:31:21 +01:00
|
|
|
return NewErrWrapper(classifyGenericError, TopLevelOperation, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func classifyOperation(ew *ErrWrapper, operation string) string {
|
|
|
|
// Basically, as explained in ErrWrapper docs, let's
|
|
|
|
// keep the child major operation, if any.
|
|
|
|
//
|
|
|
|
// QUIRK: this code is legacy code and we should not change
|
|
|
|
// it unless we also change the experiments that depend on it
|
|
|
|
// for determining the blocking reason based on the failed
|
|
|
|
// operation value (e.g., telegram, web connectivity).
|
|
|
|
if ew.Operation == ConnectOperation {
|
|
|
|
return ew.Operation
|
|
|
|
}
|
|
|
|
if ew.Operation == HTTPRoundTripOperation {
|
|
|
|
return ew.Operation
|
|
|
|
}
|
|
|
|
if ew.Operation == ResolveOperation {
|
|
|
|
return ew.Operation
|
|
|
|
}
|
|
|
|
if ew.Operation == TLSHandshakeOperation {
|
|
|
|
return ew.Operation
|
|
|
|
}
|
|
|
|
if ew.Operation == QUICHandshakeOperation {
|
|
|
|
return ew.Operation
|
|
|
|
}
|
|
|
|
if ew.Operation == "quic_handshake_start" {
|
|
|
|
return QUICHandshakeOperation
|
|
|
|
}
|
|
|
|
if ew.Operation == "quic_handshake_done" {
|
|
|
|
return QUICHandshakeOperation
|
|
|
|
}
|
|
|
|
return operation
|
2021-09-27 14:55:47 +02:00
|
|
|
}
|