-
Notifications
You must be signed in to change notification settings - Fork 476
/
Copy pathconnection_errors.go
118 lines (105 loc) · 2.95 KB
/
connection_errors.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
// Copyright 2022 CloudWeGo Authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package netpoll
import (
"fmt"
"net"
"syscall"
)
// extends syscall.Errno, the range is set to 0x100-0x1FF
const (
// The connection closed when in use.
ErrConnClosed = syscall.Errno(0x101)
// Read I/O buffer timeout, calling by Connection.Reader
ErrReadTimeout = syscall.Errno(0x102)
// Dial timeout
ErrDialTimeout = syscall.Errno(0x103)
// Calling dialer without timeout.
ErrDialNoDeadline = syscall.Errno(0x104) // TODO: no-deadline support in future
// The calling function not support.
ErrUnsupported = syscall.Errno(0x105)
// Same as io.EOF
ErrEOF = syscall.Errno(0x106)
// Write I/O buffer timeout, calling by Connection.Writer
ErrWriteTimeout = syscall.Errno(0x107)
// Concurrent connection access error
ErrConcurrentAccess = syscall.Errno(0x108)
)
const ErrnoMask = 0xFF
// wrap Errno, implement xerrors.Wrapper
func Exception(err error, suffix string) error {
no, ok := err.(syscall.Errno)
if !ok {
if suffix == "" {
return err
}
return fmt.Errorf("%s %s", err.Error(), suffix)
}
return &exception{no: no, suffix: suffix}
}
var _ net.Error = (*exception)(nil)
type exception struct {
no syscall.Errno
suffix string
}
func (e *exception) Error() string {
var s string
if int(e.no)&0x100 != 0 {
s = errnos[int(e.no)&ErrnoMask]
}
if s == "" {
s = e.no.Error()
}
if e.suffix != "" {
s += " " + e.suffix
}
return s
}
func (e *exception) Is(target error) bool {
if e == target {
return true
}
if e.no == target {
return true
}
// TODO: ErrConnClosed contains ErrEOF
if e.no == ErrEOF && target == ErrConnClosed {
return true
}
return e.no.Is(target)
}
func (e *exception) Unwrap() error {
return e.no
}
func (e *exception) Timeout() bool {
switch e.no {
case ErrDialTimeout, ErrReadTimeout, ErrWriteTimeout:
return true
}
return e.no.Timeout()
}
func (e *exception) Temporary() bool {
return e.no.Temporary()
}
// Errors defined in netpoll
var errnos = [...]string{
ErrnoMask & ErrConnClosed: "connection has been closed",
ErrnoMask & ErrReadTimeout: "connection read timeout",
ErrnoMask & ErrDialTimeout: "dial wait timeout",
ErrnoMask & ErrDialNoDeadline: "dial no deadline",
ErrnoMask & ErrUnsupported: "netpoll does not support",
ErrnoMask & ErrEOF: "EOF",
ErrnoMask & ErrWriteTimeout: "connection write timeout",
ErrnoMask & ErrConcurrentAccess: "concurrent connection access",
}