github.com/daeuniverse/quic-go@v0.0.0-20240413031024-943f218e0810/internal/qerr/errors.go (about)

     1  package qerr
     2  
     3  import (
     4  	"fmt"
     5  	"net"
     6  
     7  	"github.com/daeuniverse/quic-go/internal/protocol"
     8  )
     9  
    10  var (
    11  	ErrHandshakeTimeout = &HandshakeTimeoutError{}
    12  	ErrIdleTimeout      = &IdleTimeoutError{}
    13  )
    14  
    15  type TransportError struct {
    16  	Remote       bool
    17  	FrameType    uint64
    18  	ErrorCode    TransportErrorCode
    19  	ErrorMessage string
    20  	error        error // only set for local errors, sometimes
    21  }
    22  
    23  var _ error = &TransportError{}
    24  
    25  // NewLocalCryptoError create a new TransportError instance for a crypto error
    26  func NewLocalCryptoError(tlsAlert uint8, err error) *TransportError {
    27  	return &TransportError{
    28  		ErrorCode: 0x100 + TransportErrorCode(tlsAlert),
    29  		error:     err,
    30  	}
    31  }
    32  
    33  func (e *TransportError) Error() string {
    34  	str := fmt.Sprintf("%s (%s)", e.ErrorCode.String(), getRole(e.Remote))
    35  	if e.FrameType != 0 {
    36  		str += fmt.Sprintf(" (frame type: %#x)", e.FrameType)
    37  	}
    38  	msg := e.ErrorMessage
    39  	if len(msg) == 0 && e.error != nil {
    40  		msg = e.error.Error()
    41  	}
    42  	if len(msg) == 0 {
    43  		msg = e.ErrorCode.Message()
    44  	}
    45  	if len(msg) == 0 {
    46  		return str
    47  	}
    48  	return str + ": " + msg
    49  }
    50  
    51  func (e *TransportError) Is(target error) bool {
    52  	return target == net.ErrClosed
    53  }
    54  
    55  func (e *TransportError) Unwrap() error {
    56  	return e.error
    57  }
    58  
    59  // An ApplicationErrorCode is an application-defined error code.
    60  type ApplicationErrorCode uint64
    61  
    62  func (e *ApplicationError) Is(target error) bool {
    63  	return target == net.ErrClosed
    64  }
    65  
    66  // A StreamErrorCode is an error code used to cancel streams.
    67  type StreamErrorCode uint64
    68  
    69  type ApplicationError struct {
    70  	Remote       bool
    71  	ErrorCode    ApplicationErrorCode
    72  	ErrorMessage string
    73  }
    74  
    75  var _ error = &ApplicationError{}
    76  
    77  func (e *ApplicationError) Error() string {
    78  	if len(e.ErrorMessage) == 0 {
    79  		return fmt.Sprintf("Application error %#x (%s)", e.ErrorCode, getRole(e.Remote))
    80  	}
    81  	return fmt.Sprintf("Application error %#x (%s): %s", e.ErrorCode, getRole(e.Remote), e.ErrorMessage)
    82  }
    83  
    84  type IdleTimeoutError struct{}
    85  
    86  var _ error = &IdleTimeoutError{}
    87  
    88  func (e *IdleTimeoutError) Timeout() bool        { return true }
    89  func (e *IdleTimeoutError) Temporary() bool      { return false }
    90  func (e *IdleTimeoutError) Error() string        { return "timeout: no recent network activity" }
    91  func (e *IdleTimeoutError) Is(target error) bool { return target == net.ErrClosed }
    92  
    93  type HandshakeTimeoutError struct{}
    94  
    95  var _ error = &HandshakeTimeoutError{}
    96  
    97  func (e *HandshakeTimeoutError) Timeout() bool        { return true }
    98  func (e *HandshakeTimeoutError) Temporary() bool      { return false }
    99  func (e *HandshakeTimeoutError) Error() string        { return "timeout: handshake did not complete in time" }
   100  func (e *HandshakeTimeoutError) Is(target error) bool { return target == net.ErrClosed }
   101  
   102  // A VersionNegotiationError occurs when the client and the server can't agree on a QUIC version.
   103  type VersionNegotiationError struct {
   104  	Ours   []protocol.Version
   105  	Theirs []protocol.Version
   106  }
   107  
   108  func (e *VersionNegotiationError) Error() string {
   109  	return fmt.Sprintf("no compatible QUIC version found (we support %s, server offered %s)", e.Ours, e.Theirs)
   110  }
   111  
   112  func (e *VersionNegotiationError) Is(target error) bool {
   113  	return target == net.ErrClosed
   114  }
   115  
   116  // A StatelessResetError occurs when we receive a stateless reset.
   117  type StatelessResetError struct {
   118  	Token protocol.StatelessResetToken
   119  }
   120  
   121  var _ net.Error = &StatelessResetError{}
   122  
   123  func (e *StatelessResetError) Error() string {
   124  	return fmt.Sprintf("received a stateless reset with token %x", e.Token)
   125  }
   126  
   127  func (e *StatelessResetError) Is(target error) bool {
   128  	return target == net.ErrClosed
   129  }
   130  
   131  func (e *StatelessResetError) Timeout() bool   { return false }
   132  func (e *StatelessResetError) Temporary() bool { return true }
   133  
   134  func getRole(remote bool) string {
   135  	if remote {
   136  		return "remote"
   137  	}
   138  	return "local"
   139  }