expose a VersionNegoationError

This commit is contained in:
Marten Seemann 2021-04-25 19:03:34 +07:00
parent 42b61729bd
commit 1ce572228b
5 changed files with 47 additions and 24 deletions

View file

@ -1,10 +1,16 @@
package quic package quic
import "github.com/lucas-clemente/quic-go/internal/qerr" import (
"github.com/lucas-clemente/quic-go/internal/qerr"
)
type (
TransportError = qerr.TransportError
ApplicationError = qerr.ApplicationError
VersionNegotiationError = qerr.VersionNegotiationError
)
type ( type (
TransportError = qerr.TransportError
ApplicationError = qerr.ApplicationError
TransportErrorCode = qerr.TransportErrorCode TransportErrorCode = qerr.TransportErrorCode
ApplicationErrorCode = qerr.ApplicationErrorCode ApplicationErrorCode = qerr.ApplicationErrorCode
) )

View file

@ -371,7 +371,7 @@ var _ = Describe("MITM test", func() {
} }
err := runTest(delayCb) err := runTest(delayCb)
Expect(err).To(HaveOccurred()) Expect(err).To(HaveOccurred())
Expect(err.Error()).To(ContainSubstring("no compatible QUIC version found")) Expect(err).To(MatchError(&quic.VersionNegotiationError{}))
}) })
// times out, because client doesn't accept subsequent real retry packets from server // times out, because client doesn't accept subsequent real retry packets from server

View file

@ -2,6 +2,8 @@ package qerr
import ( import (
"fmt" "fmt"
"github.com/lucas-clemente/quic-go/internal/protocol"
) )
var ( var (
@ -92,3 +94,18 @@ func (e *HandshakeTimeoutError) Is(target error) bool {
_, ok := target.(*HandshakeTimeoutError) _, ok := target.(*HandshakeTimeoutError)
return ok return ok
} }
// A VersionNegotiationError occurs when the client and the server can't agree on a QUIC version.
type VersionNegotiationError struct {
Ours []protocol.VersionNumber
Theirs []protocol.VersionNumber
}
func (e *VersionNegotiationError) Error() string {
return fmt.Sprintf("no compatible QUIC version found (we support %s, server offered %s)", e.Ours, e.Theirs)
}
func (e *VersionNegotiationError) Is(target error) bool {
_, ok := target.(*VersionNegotiationError)
return ok
}

View file

@ -4,6 +4,7 @@ import (
"errors" "errors"
"net" "net"
"github.com/lucas-clemente/quic-go/internal/protocol"
. "github.com/onsi/ginkgo" . "github.com/onsi/ginkgo"
. "github.com/onsi/gomega" . "github.com/onsi/gomega"
) )
@ -101,4 +102,17 @@ var _ = Describe("QUIC Errors", func() {
Expect(errors.Is(err, &IdleTimeoutError{})).To(BeTrue()) Expect(errors.Is(err, &IdleTimeoutError{})).To(BeTrue())
}) })
}) })
Context("Version Negotiation errors", func() {
It("is a Version Negotiation error", func() {
Expect(errors.Is(&VersionNegotiationError{Ours: []protocol.VersionNumber{2, 3}}, &VersionNegotiationError{})).To(BeTrue())
})
It("has a string representation", func() {
Expect((&VersionNegotiationError{
Ours: []protocol.VersionNumber{2, 3},
Theirs: []protocol.VersionNumber{4, 5, 6},
}).Error()).To(Equal("no compatible QUIC version found (we support [0x2 0x3], server offered [0x4 0x5 0x6])"))
})
})
}) })

View file

@ -132,20 +132,6 @@ func (e *errCloseForRecreating) Is(target error) bool {
return ok return ok
} }
type errVersionNegotiation struct {
ourVersions []protocol.VersionNumber
theirVersions []protocol.VersionNumber
}
func (e *errVersionNegotiation) Error() string {
return fmt.Sprintf("no compatible QUIC version found (we support %s, server offered %s)", e.ourVersions, e.theirVersions)
}
func (e *errVersionNegotiation) Is(target error) bool {
_, ok := target.(*errVersionNegotiation)
return ok
}
var sessionTracingID uint64 // to be accessed atomically var sessionTracingID uint64 // to be accessed atomically
func nextSessionTracingID() uint64 { return atomic.AddUint64(&sessionTracingID, 1) } func nextSessionTracingID() uint64 { return atomic.AddUint64(&sessionTracingID, 1) }
@ -1101,9 +1087,9 @@ func (s *session) handleVersionNegotiationPacket(p *receivedPacket) {
} }
newVersion, ok := protocol.ChooseSupportedVersion(s.config.Versions, supportedVersions) newVersion, ok := protocol.ChooseSupportedVersion(s.config.Versions, supportedVersions)
if !ok { if !ok {
s.destroyImpl(&errVersionNegotiation{ s.destroyImpl(&VersionNegotiationError{
ourVersions: s.config.Versions, Ours: s.config.Versions,
theirVersions: supportedVersions, Theirs: supportedVersions,
}) })
s.logger.Infof("No compatible QUIC version found.") s.logger.Infof("No compatible QUIC version found.")
return return
@ -1498,7 +1484,7 @@ func (s *session) handleCloseError(closeErr *closeError) {
case errors.Is(e, qerr.ErrIdleTimeout), case errors.Is(e, qerr.ErrIdleTimeout),
errors.Is(e, qerr.ErrHandshakeTimeout), errors.Is(e, qerr.ErrHandshakeTimeout),
errors.Is(e, &statelessResetErr{}), errors.Is(e, &statelessResetErr{}),
errors.Is(e, &errVersionNegotiation{}), errors.Is(e, &VersionNegotiationError{}),
errors.Is(e, &errCloseForRecreating{}), errors.Is(e, &errCloseForRecreating{}),
errors.Is(e, &qerr.ApplicationError{}), errors.Is(e, &qerr.ApplicationError{}),
errors.Is(e, &qerr.TransportError{}): errors.Is(e, &qerr.TransportError{}):
@ -1518,7 +1504,7 @@ func (s *session) handleCloseError(closeErr *closeError) {
if s.tracer != nil && !errors.Is(e, &errCloseForRecreating{}) { if s.tracer != nil && !errors.Is(e, &errCloseForRecreating{}) {
var ( var (
resetErr *statelessResetErr resetErr *statelessResetErr
vnErr *errVersionNegotiation vnErr *VersionNegotiationError
transportErr *qerr.TransportError transportErr *qerr.TransportError
applicationErr *qerr.ApplicationError applicationErr *qerr.ApplicationError
) )
@ -1530,7 +1516,7 @@ func (s *session) handleCloseError(closeErr *closeError) {
case errors.As(e, &resetErr): case errors.As(e, &resetErr):
s.tracer.ClosedConnection(logging.NewStatelessResetCloseReason(resetErr.token)) s.tracer.ClosedConnection(logging.NewStatelessResetCloseReason(resetErr.token))
case errors.As(e, &vnErr): case errors.As(e, &vnErr):
s.tracer.ClosedConnection(logging.NewVersionNegotiationError(vnErr.theirVersions)) s.tracer.ClosedConnection(logging.NewVersionNegotiationError(vnErr.Theirs))
case errors.As(e, &applicationErr): case errors.As(e, &applicationErr):
s.tracer.ClosedConnection(logging.NewApplicationCloseReason(logging.ApplicationError(applicationErr.ErrorCode), closeErr.remote)) s.tracer.ClosedConnection(logging.NewApplicationCloseReason(logging.ApplicationError(applicationErr.ErrorCode), closeErr.remote))
case errors.As(e, &transportErr): case errors.As(e, &transportErr):