refactor(netxlite/errorsx): change all tests to be unit tests (#483)

Later we will try to write comprehensive integration tests for
the whole netxlite package. We want just unit tests here.
This commit is contained in:
Simone Basso 2021-09-07 22:10:29 +02:00 committed by GitHub
parent b7786a7324
commit 1472f7530b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 122 additions and 124 deletions

View File

@ -124,19 +124,6 @@ const (
quicTLSUnrecognizedName = 112 quicTLSUnrecognizedName = 112
) )
// quicIsCertificateError tells us whether a specific TLS alert error
// we received is actually an error depending on the certificate.
//
// The set of checks we implement here is a set of heuristics based
// on our understanding of the TLS spec and may need tweaks.
func quicIsCertificateError(alert uint8) bool {
return (alert == quicTLSAlertBadCertificate ||
alert == quicTLSAlertUnsupportedCertificate ||
alert == quicTLSAlertCertificateExpired ||
alert == quicTLSAlertCertificateRevoked ||
alert == quicTLSAlertCertificateUnknown)
}
// ClassifyQUICHandshakeError maps an error occurred during the QUIC // ClassifyQUICHandshakeError maps an error occurred during the QUIC
// handshake to an OONI failure string. // handshake to an OONI failure string.
// //
@ -196,6 +183,29 @@ func ClassifyQUICHandshakeError(err error) string {
return ClassifyGenericError(err) return ClassifyGenericError(err)
} }
// quicIsCertificateError tells us whether a specific TLS alert error
// we received is actually an error depending on the certificate.
//
// The set of checks we implement here is a set of heuristics based
// on our understanding of the TLS spec and may need tweaks.
func quicIsCertificateError(alert uint8) bool {
// List out each case separately so we know we test them
switch alert {
case quicTLSAlertBadCertificate:
return true
case quicTLSAlertUnsupportedCertificate:
return true
case quicTLSAlertCertificateExpired:
return true
case quicTLSAlertCertificateRevoked:
return true
case quicTLSAlertCertificateUnknown:
return true
default:
return false
}
}
// ErrDNSBogon indicates that we found a bogon address. Code that // ErrDNSBogon indicates that we found a bogon address. Code that
// filters for DNS bogons MUST use this error. // filters for DNS bogons MUST use this error.
var ErrDNSBogon = errors.New("dns: detected bogon address") var ErrDNSBogon = errors.New("dns: detected bogon address")

View File

@ -2,12 +2,9 @@ package errorsx
import ( import (
"context" "context"
"crypto/tls"
"crypto/x509" "crypto/x509"
"errors" "errors"
"io" "io"
"net"
"syscall"
"testing" "testing"
"github.com/google/go-cmp/cmp" "github.com/google/go-cmp/cmp"
@ -16,6 +13,9 @@ import (
) )
func TestClassifyGenericError(t *testing.T) { func TestClassifyGenericError(t *testing.T) {
// Please, keep this list sorted in the same order
// in which checks appear on the code
t.Run("for input being already an ErrWrapper", func(t *testing.T) { t.Run("for input being already an ErrWrapper", func(t *testing.T) {
err := &ErrWrapper{Failure: FailureEOFError} err := &ErrWrapper{Failure: FailureEOFError}
if ClassifyGenericError(err) != FailureEOFError { if ClassifyGenericError(err) != FailureEOFError {
@ -23,20 +23,18 @@ func TestClassifyGenericError(t *testing.T) {
} }
}) })
t.Run("for already wrapped error", func(t *testing.T) { t.Run("for a system call error", func(t *testing.T) {
err := io.EOF if ClassifyGenericError(EWOULDBLOCK) != FailureOperationWouldBlock {
if ClassifyGenericError(err) != FailureEOFError { t.Fatal("unexpected results")
t.Fatal("unexpected result")
} }
}) })
t.Run("for context.Canceled", func(t *testing.T) { // Now we enter into classifyWithStringSuffix. We test it here
if ClassifyGenericError(context.Canceled) != FailureInterrupted { // since we want to test the ClassifyGenericError in is
t.Fatal("unexpected result") // entirety here and the classifyWithStringSuffix function
} // is just an implementation detail.
})
t.Run("for operation was canceled error", func(t *testing.T) { t.Run("for operation was canceled", func(t *testing.T) {
if ClassifyGenericError(errors.New("operation was canceled")) != FailureInterrupted { if ClassifyGenericError(errors.New("operation was canceled")) != FailureInterrupted {
t.Fatal("unexpected result") t.Fatal("unexpected result")
} }
@ -44,45 +42,12 @@ func TestClassifyGenericError(t *testing.T) {
t.Run("for EOF", func(t *testing.T) { t.Run("for EOF", func(t *testing.T) {
if ClassifyGenericError(io.EOF) != FailureEOFError { if ClassifyGenericError(io.EOF) != FailureEOFError {
t.Fatal("unexpected results") t.Fatal("unexpected result")
}
})
t.Run("for canceled", func(t *testing.T) {
if ClassifyGenericError(syscall.ECANCELED) != FailureOperationCanceled {
t.Fatal("unexpected results")
}
})
t.Run("for connection_refused", func(t *testing.T) {
if ClassifyGenericError(syscall.ECONNREFUSED) != FailureConnectionRefused {
t.Fatal("unexpected results")
}
})
t.Run("for connection_reset", func(t *testing.T) {
if ClassifyGenericError(syscall.ECONNRESET) != FailureConnectionReset {
t.Fatal("unexpected results")
}
})
t.Run("for host_unreachable", func(t *testing.T) {
if ClassifyGenericError(syscall.EHOSTUNREACH) != FailureHostUnreachable {
t.Fatal("unexpected results")
}
})
t.Run("for system timeout", func(t *testing.T) {
if ClassifyGenericError(syscall.ETIMEDOUT) != FailureTimedOut {
t.Fatal("unexpected results")
} }
}) })
t.Run("for context deadline exceeded", func(t *testing.T) { t.Run("for context deadline exceeded", func(t *testing.T) {
ctx, cancel := context.WithTimeout(context.Background(), 1) if ClassifyGenericError(context.DeadlineExceeded) != FailureGenericTimeoutError {
defer cancel()
<-ctx.Done()
if ClassifyGenericError(ctx.Err()) != FailureGenericTimeoutError {
t.Fatal("unexpected results") t.Fatal("unexpected results")
} }
}) })
@ -93,22 +58,13 @@ func TestClassifyGenericError(t *testing.T) {
} }
}) })
t.Run("for i/o error", func(t *testing.T) { t.Run("for i/o timeout", func(t *testing.T) {
ctx, cancel := context.WithTimeout(context.Background(), 1) if ClassifyGenericError(errors.New("i/o timeout")) != FailureGenericTimeoutError {
defer cancel() // fail immediately
conn, err := (&net.Dialer{}).DialContext(ctx, "tcp", "www.google.com:80")
if err == nil {
t.Fatal("expected an error here")
}
if conn != nil {
t.Fatal("expected nil connection here")
}
if ClassifyGenericError(err) != FailureGenericTimeoutError {
t.Fatal("unexpected results") t.Fatal("unexpected results")
} }
}) })
t.Run("for TLS handshake timeout error", func(t *testing.T) { t.Run("for TLS handshake timeout", func(t *testing.T) {
err := errors.New("net/http: TLS handshake timeout") err := errors.New("net/http: TLS handshake timeout")
if ClassifyGenericError(err) != FailureGenericTimeoutError { if ClassifyGenericError(err) != FailureGenericTimeoutError {
t.Fatal("unexpected results") t.Fatal("unexpected results")
@ -116,53 +72,44 @@ func TestClassifyGenericError(t *testing.T) {
}) })
t.Run("for no such host", func(t *testing.T) { t.Run("for no such host", func(t *testing.T) {
if ClassifyGenericError(&net.DNSError{ if ClassifyGenericError(errors.New("no such host")) != FailureDNSNXDOMAINError {
Err: "no such host",
}) != FailureDNSNXDOMAINError {
t.Fatal("unexpected results") t.Fatal("unexpected results")
} }
}) })
t.Run("for errors including IPv4 address", func(t *testing.T) { // Now we're back in ClassifyGenericError
input := errors.New("read tcp 10.0.2.15:56948->93.184.216.34:443: use of closed network connection")
expected := "unknown_failure: read tcp [scrubbed]->[scrubbed]: use of closed network connection" t.Run("for context.Canceled", func(t *testing.T) {
out := ClassifyGenericError(input) if ClassifyGenericError(context.Canceled) != FailureInterrupted {
if out != expected { t.Fatal("unexpected result")
t.Fatal(cmp.Diff(expected, out))
} }
}) })
t.Run("for errors including IPv6 address", func(t *testing.T) { t.Run("for unknown errors", func(t *testing.T) {
input := errors.New("read tcp [::1]:56948->[::1]:443: use of closed network connection") t.Run("with an IPv4 address", func(t *testing.T) {
expected := "unknown_failure: read tcp [scrubbed]->[scrubbed]: use of closed network connection" input := errors.New("read tcp 10.0.2.15:56948->93.184.216.34:443: use of closed network connection")
out := ClassifyGenericError(input) expected := "unknown_failure: read tcp [scrubbed]->[scrubbed]: use of closed network connection"
if out != expected { out := ClassifyGenericError(input)
t.Fatal(cmp.Diff(expected, out)) if out != expected {
} t.Fatal(cmp.Diff(expected, out))
}) }
})
t.Run("for i/o error", func(t *testing.T) { t.Run("with an IPv6 address", func(t *testing.T) {
ctx, cancel := context.WithTimeout(context.Background(), 1) input := errors.New("read tcp [::1]:56948->[::1]:443: use of closed network connection")
defer cancel() // fail immediately expected := "unknown_failure: read tcp [scrubbed]->[scrubbed]: use of closed network connection"
udpAddr := &net.UDPAddr{IP: net.ParseIP("216.58.212.164"), Port: 80, Zone: ""} out := ClassifyGenericError(input)
udpConn, err := net.ListenUDP("udp", &net.UDPAddr{IP: net.IPv4zero, Port: 0}) if out != expected {
if err != nil { t.Fatal(cmp.Diff(expected, out))
t.Fatal(err) }
} })
sess, err := quic.DialEarlyContext(ctx, udpConn, udpAddr, "google.com:80", &tls.Config{}, &quic.Config{})
if err == nil {
t.Fatal("expected an error here")
}
if sess != nil {
t.Fatal("expected nil session here")
}
if ClassifyGenericError(err) != FailureGenericTimeoutError {
t.Fatal("unexpected results")
}
}) })
} }
func TestClassifyQUICHandshakeError(t *testing.T) { func TestClassifyQUICHandshakeError(t *testing.T) {
// Please, keep this list sorted in the same order
// in which checks appear on the code
t.Run("for input being already an ErrWrapper", func(t *testing.T) { t.Run("for input being already an ErrWrapper", func(t *testing.T) {
err := &ErrWrapper{Failure: FailureEOFError} err := &ErrWrapper{Failure: FailureEOFError}
if ClassifyQUICHandshakeError(err) != FailureEOFError { if ClassifyQUICHandshakeError(err) != FailureEOFError {
@ -170,58 +117,93 @@ func TestClassifyQUICHandshakeError(t *testing.T) {
} }
}) })
t.Run("for connection_reset", func(t *testing.T) {
if ClassifyQUICHandshakeError(&quic.StatelessResetError{}) != FailureConnectionReset {
t.Fatal("unexpected results")
}
})
t.Run("for incompatible quic version", func(t *testing.T) { t.Run("for incompatible quic version", func(t *testing.T) {
if ClassifyQUICHandshakeError(&quic.VersionNegotiationError{}) != FailureQUICIncompatibleVersion { if ClassifyQUICHandshakeError(&quic.VersionNegotiationError{}) != FailureQUICIncompatibleVersion {
t.Fatal("unexpected results") t.Fatal("unexpected results")
} }
}) })
t.Run("for quic connection refused", func(t *testing.T) { t.Run("for stateless reset", func(t *testing.T) {
if ClassifyQUICHandshakeError(&quic.TransportError{ErrorCode: quic.ConnectionRefused}) != FailureConnectionRefused { if ClassifyQUICHandshakeError(&quic.StatelessResetError{}) != FailureConnectionReset {
t.Fatal("unexpected results") t.Fatal("unexpected results")
} }
}) })
t.Run("for quic handshake timeout", func(t *testing.T) { t.Run("for handshake timeout", func(t *testing.T) {
if ClassifyQUICHandshakeError(&quic.HandshakeTimeoutError{}) != FailureGenericTimeoutError { if ClassifyQUICHandshakeError(&quic.HandshakeTimeoutError{}) != FailureGenericTimeoutError {
t.Fatal("unexpected results") t.Fatal("unexpected results")
} }
}) })
t.Run("for QUIC idle connection timeout", func(t *testing.T) { t.Run("for idle timeout", func(t *testing.T) {
if ClassifyQUICHandshakeError(&quic.IdleTimeoutError{}) != FailureGenericTimeoutError { if ClassifyQUICHandshakeError(&quic.IdleTimeoutError{}) != FailureGenericTimeoutError {
t.Fatal("unexpected results") t.Fatal("unexpected results")
} }
}) })
t.Run("for QUIC CRYPTO Handshake", func(t *testing.T) { t.Run("for connection refused", func(t *testing.T) {
var err quic.TransportErrorCode = quicTLSAlertHandshakeFailure if ClassifyQUICHandshakeError(&quic.TransportError{ErrorCode: quic.ConnectionRefused}) != FailureConnectionRefused {
if ClassifyQUICHandshakeError(&quic.TransportError{ErrorCode: err}) != FailureSSLFailedHandshake {
t.Fatal("unexpected results") t.Fatal("unexpected results")
} }
}) })
t.Run("for QUIC CRYPTO Invalid Certificate", func(t *testing.T) { t.Run("for bad certificate", func(t *testing.T) {
var err quic.TransportErrorCode = quicTLSAlertBadCertificate var err quic.TransportErrorCode = quicTLSAlertBadCertificate
if ClassifyQUICHandshakeError(&quic.TransportError{ErrorCode: err}) != FailureSSLInvalidCertificate { if ClassifyQUICHandshakeError(&quic.TransportError{ErrorCode: err}) != FailureSSLInvalidCertificate {
t.Fatal("unexpected results") t.Fatal("unexpected results")
} }
}) })
t.Run("for QUIC CRYPTO Unknown CA", func(t *testing.T) { t.Run("for unsupported certificate", func(t *testing.T) {
var err quic.TransportErrorCode = quicTLSAlertUnsupportedCertificate
if ClassifyQUICHandshakeError(&quic.TransportError{ErrorCode: err}) != FailureSSLInvalidCertificate {
t.Fatal("unexpected results")
}
})
t.Run("for certificate expired", func(t *testing.T) {
var err quic.TransportErrorCode = quicTLSAlertCertificateExpired
if ClassifyQUICHandshakeError(&quic.TransportError{ErrorCode: err}) != FailureSSLInvalidCertificate {
t.Fatal("unexpected results")
}
})
t.Run("for certificate revoked", func(t *testing.T) {
var err quic.TransportErrorCode = quicTLSAlertCertificateRevoked
if ClassifyQUICHandshakeError(&quic.TransportError{ErrorCode: err}) != FailureSSLInvalidCertificate {
t.Fatal("unexpected results")
}
})
t.Run("for certificate unknown", func(t *testing.T) {
var err quic.TransportErrorCode = quicTLSAlertCertificateUnknown
if ClassifyQUICHandshakeError(&quic.TransportError{ErrorCode: err}) != FailureSSLInvalidCertificate {
t.Fatal("unexpected results")
}
})
t.Run("for decrypt error", func(t *testing.T) {
var err quic.TransportErrorCode = quicTLSAlertDecryptError
if ClassifyQUICHandshakeError(&quic.TransportError{ErrorCode: err}) != FailureSSLFailedHandshake {
t.Fatal("unexpected results")
}
})
t.Run("for handshake failure", func(t *testing.T) {
var err quic.TransportErrorCode = quicTLSAlertHandshakeFailure
if ClassifyQUICHandshakeError(&quic.TransportError{ErrorCode: err}) != FailureSSLFailedHandshake {
t.Fatal("unexpected results")
}
})
t.Run("for unknown CA", func(t *testing.T) {
var err quic.TransportErrorCode = quicTLSAlertUnknownCA var err quic.TransportErrorCode = quicTLSAlertUnknownCA
if ClassifyQUICHandshakeError(&quic.TransportError{ErrorCode: err}) != FailureSSLUnknownAuthority { if ClassifyQUICHandshakeError(&quic.TransportError{ErrorCode: err}) != FailureSSLUnknownAuthority {
t.Fatal("unexpected results") t.Fatal("unexpected results")
} }
}) })
t.Run("for QUIC CRYPTO Bad Hostname", func(t *testing.T) { t.Run("for unrecognized hostname", func(t *testing.T) {
var err quic.TransportErrorCode = quicTLSUnrecognizedName var err quic.TransportErrorCode = quicTLSUnrecognizedName
if ClassifyQUICHandshakeError(&quic.TransportError{ErrorCode: err}) != FailureSSLInvalidHostname { if ClassifyQUICHandshakeError(&quic.TransportError{ErrorCode: err}) != FailureSSLInvalidHostname {
t.Fatal("unexpected results") t.Fatal("unexpected results")
@ -236,6 +218,9 @@ func TestClassifyQUICHandshakeError(t *testing.T) {
} }
func TestClassifyResolverError(t *testing.T) { func TestClassifyResolverError(t *testing.T) {
// Please, keep this list sorted in the same order
// in which checks appear on the code
t.Run("for input being already an ErrWrapper", func(t *testing.T) { t.Run("for input being already an ErrWrapper", func(t *testing.T) {
err := &ErrWrapper{Failure: FailureEOFError} err := &ErrWrapper{Failure: FailureEOFError}
if ClassifyResolverError(err) != FailureEOFError { if ClassifyResolverError(err) != FailureEOFError {
@ -257,6 +242,9 @@ func TestClassifyResolverError(t *testing.T) {
} }
func TestClassifyTLSHandshakeError(t *testing.T) { func TestClassifyTLSHandshakeError(t *testing.T) {
// Please, keep this list sorted in the same order
// in which checks appear on the code
t.Run("for input being already an ErrWrapper", func(t *testing.T) { t.Run("for input being already an ErrWrapper", func(t *testing.T) {
err := &ErrWrapper{Failure: FailureEOFError} err := &ErrWrapper{Failure: FailureEOFError}
if ClassifyTLSHandshakeError(err) != FailureEOFError { if ClassifyTLSHandshakeError(err) != FailureEOFError {