mirror of
https://github.com/SagerNet/sing.git
synced 2025-04-04 20:37:40 +03:00
Add multi error
This commit is contained in:
parent
f00396c60e
commit
f8356c256f
10 changed files with 141 additions and 102 deletions
34
.github/workflows/linter.yml
vendored
34
.github/workflows/linter.yml
vendored
|
@ -1,34 +0,0 @@
|
||||||
name: Linter
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- dev
|
|
||||||
paths:
|
|
||||||
- "**/*.go"
|
|
||||||
- ".github/workflows/linter.yml"
|
|
||||||
pull_request:
|
|
||||||
types: [ opened, synchronize, reopened ]
|
|
||||||
paths:
|
|
||||||
- "**/*.go"
|
|
||||||
- ".github/workflows/linter.yml"
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
lint:
|
|
||||||
if: github.repository == 'sagernet/sing'
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v2
|
|
||||||
- name: Get latest go version
|
|
||||||
id: version
|
|
||||||
run: |
|
|
||||||
echo ::set-output name=go_version::$(curl -s https://raw.githubusercontent.com/actions/go-versions/main/versions-manifest.json | grep -oE '"version": "[0-9]{1}.[0-9]{1,}(.[0-9]{1,})?"' | head -1 | cut -d':' -f2 | sed 's/ //g; s/"//g')
|
|
||||||
- name: Setup Go
|
|
||||||
uses: actions/setup-go@v2
|
|
||||||
with:
|
|
||||||
go-version: ${{ steps.version.outputs.go_version }}
|
|
||||||
- name: golangci-lint
|
|
||||||
uses: golangci/golangci-lint-action@v3
|
|
||||||
with:
|
|
||||||
version: latest
|
|
|
@ -78,6 +78,20 @@ func Filter[T any](arr []T, block func(it T) bool) []T {
|
||||||
return retArr
|
return retArr
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func FilterNotNil[T any](arr []T) []T {
|
||||||
|
return Filter(arr, func(it T) bool {
|
||||||
|
var anyIt any = it
|
||||||
|
return anyIt != nil
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func FilterNotDefault[T comparable](arr []T) []T {
|
||||||
|
var defaultValue T
|
||||||
|
return Filter(arr, func(it T) bool {
|
||||||
|
return it != defaultValue
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
func Find[T any](arr []T, block func(it T) bool) T {
|
func Find[T any](arr []T, block func(it T) bool) T {
|
||||||
for _, it := range arr {
|
for _, it := range arr {
|
||||||
if block(it) {
|
if block(it) {
|
||||||
|
@ -163,6 +177,7 @@ func Must2(_, _ any, err error) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Deprecated: use E.Errors
|
||||||
func AnyError(errs ...error) error {
|
func AnyError(errs ...error) error {
|
||||||
for _, err := range errs {
|
for _, err := range errs {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
17
common/exceptions/cause.go
Normal file
17
common/exceptions/cause.go
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
package exceptions
|
||||||
|
|
||||||
|
type causeError struct {
|
||||||
|
message string
|
||||||
|
cause error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *causeError) Error() string {
|
||||||
|
if e.cause == nil {
|
||||||
|
return e.message
|
||||||
|
}
|
||||||
|
return e.message + ": " + e.cause.Error()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *causeError) Unwrap() error {
|
||||||
|
return e.cause
|
||||||
|
}
|
|
@ -8,39 +8,16 @@ import (
|
||||||
"net"
|
"net"
|
||||||
"os"
|
"os"
|
||||||
|
|
||||||
|
"github.com/sagernet/sing/common"
|
||||||
F "github.com/sagernet/sing/common/format"
|
F "github.com/sagernet/sing/common/format"
|
||||||
)
|
)
|
||||||
|
|
||||||
type causeError struct {
|
type Handler interface {
|
||||||
message string
|
NewError(ctx context.Context, err error)
|
||||||
cause error
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *causeError) Error() string {
|
type MultiError interface {
|
||||||
if e.cause == nil {
|
UnwrapMulti() []error
|
||||||
return e.message
|
|
||||||
}
|
|
||||||
return e.message + ": " + e.cause.Error()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *causeError) Unwrap() error {
|
|
||||||
return e.cause
|
|
||||||
}
|
|
||||||
|
|
||||||
type extendedError struct {
|
|
||||||
message string
|
|
||||||
cause error
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *extendedError) Error() string {
|
|
||||||
if e.cause == nil {
|
|
||||||
return e.message
|
|
||||||
}
|
|
||||||
return e.cause.Error() + ": " + e.message
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *extendedError) Unwrap() error {
|
|
||||||
return e.cause
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func New(message ...any) error {
|
func New(message ...any) error {
|
||||||
|
@ -55,23 +32,17 @@ func Extend(cause error, message ...any) error {
|
||||||
return &extendedError{F.ToString(message...), cause}
|
return &extendedError{F.ToString(message...), cause}
|
||||||
}
|
}
|
||||||
|
|
||||||
type HasInnerError interface {
|
func Errors(errors ...error) error {
|
||||||
Unwrap() error
|
errors = common.FilterNotNil(errors)
|
||||||
}
|
switch len(errors) {
|
||||||
|
case 0:
|
||||||
func Unwrap(err error) error {
|
return nil
|
||||||
for {
|
case 1:
|
||||||
inner, ok := err.(HasInnerError)
|
return errors[0]
|
||||||
if !ok {
|
}
|
||||||
break
|
return &multiError{
|
||||||
}
|
errors: errors,
|
||||||
innerErr := inner.Unwrap()
|
|
||||||
if innerErr == nil {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
err = innerErr
|
|
||||||
}
|
}
|
||||||
return err
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func IsCanceled(err error) bool {
|
func IsCanceled(err error) bool {
|
||||||
|
@ -81,24 +52,3 @@ func IsCanceled(err error) bool {
|
||||||
func IsClosed(err error) bool {
|
func IsClosed(err error) bool {
|
||||||
return errors.Is(err, io.EOF) || errors.Is(err, net.ErrClosed) || errors.Is(err, io.ErrClosedPipe) || errors.Is(err, os.ErrClosed)
|
return errors.Is(err, io.EOF) || errors.Is(err, net.ErrClosed) || errors.Is(err, io.ErrClosedPipe) || errors.Is(err, os.ErrClosed)
|
||||||
}
|
}
|
||||||
|
|
||||||
type TimeoutError interface {
|
|
||||||
Timeout() bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func IsTimeout(err error) bool {
|
|
||||||
if unwrapErr := errors.Unwrap(err); unwrapErr != nil {
|
|
||||||
err = unwrapErr
|
|
||||||
}
|
|
||||||
if ne, ok := err.(*os.SyscallError); ok {
|
|
||||||
err = ne.Err
|
|
||||||
}
|
|
||||||
if timeoutErr, isTimeoutErr := err.(TimeoutError); isTimeoutErr {
|
|
||||||
return timeoutErr.Timeout()
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
type Handler interface {
|
|
||||||
NewError(ctx context.Context, err error)
|
|
||||||
}
|
|
||||||
|
|
17
common/exceptions/extend.go
Normal file
17
common/exceptions/extend.go
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
package exceptions
|
||||||
|
|
||||||
|
type extendedError struct {
|
||||||
|
message string
|
||||||
|
cause error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *extendedError) Error() string {
|
||||||
|
if e.cause == nil {
|
||||||
|
return e.message
|
||||||
|
}
|
||||||
|
return e.cause.Error() + ": " + e.message
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *extendedError) Unwrap() error {
|
||||||
|
return e.cause
|
||||||
|
}
|
20
common/exceptions/inner.go
Normal file
20
common/exceptions/inner.go
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
package exceptions
|
||||||
|
|
||||||
|
type HasInnerError interface {
|
||||||
|
Unwrap() error
|
||||||
|
}
|
||||||
|
|
||||||
|
func Unwrap(err error) error {
|
||||||
|
for {
|
||||||
|
inner, ok := err.(HasInnerError)
|
||||||
|
if !ok {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
innerErr := inner.Unwrap()
|
||||||
|
if innerErr == nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
err = innerErr
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
31
common/exceptions/multi.go
Normal file
31
common/exceptions/multi.go
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
package exceptions
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/sagernet/sing/common"
|
||||||
|
F "github.com/sagernet/sing/common/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type multiError struct {
|
||||||
|
errors []error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *multiError) Error() string {
|
||||||
|
return "multi error: (" + strings.Join(common.Map(e.errors, F.ToString0[error]), " | ") + ")"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *multiError) Unwrap() error {
|
||||||
|
return e.errors[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *multiError) UnwrapMulti() []error {
|
||||||
|
return e.errors
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *multiError) Is(err error) bool {
|
||||||
|
return common.Any(e.errors, func(it error) bool {
|
||||||
|
return errors.Is(it, err)
|
||||||
|
})
|
||||||
|
}
|
23
common/exceptions/timeout.go
Normal file
23
common/exceptions/timeout.go
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
package exceptions
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"os"
|
||||||
|
)
|
||||||
|
|
||||||
|
type TimeoutError interface {
|
||||||
|
Timeout() bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func IsTimeout(err error) bool {
|
||||||
|
if unwrapErr := errors.Unwrap(err); unwrapErr != nil {
|
||||||
|
err = unwrapErr
|
||||||
|
}
|
||||||
|
if ne, ok := err.(*os.SyscallError); ok {
|
||||||
|
err = ne.Err
|
||||||
|
}
|
||||||
|
if timeoutErr, isTimeoutErr := err.(TimeoutError); isTimeoutErr {
|
||||||
|
return timeoutErr.Timeout()
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
|
@ -118,7 +118,7 @@ func HandleConnection(ctx context.Context, conn net.Conn, reader *std_bufio.Read
|
||||||
|
|
||||||
response, err := httpClient.Do(request)
|
response, err := httpClient.Do(request)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return common.AnyError(innerErr, err, responseWith(request, http.StatusBadGateway).Write(conn))
|
return E.Errors(innerErr, err, responseWith(request, http.StatusBadGateway).Write(conn))
|
||||||
}
|
}
|
||||||
|
|
||||||
removeHopByHopHeaders(response.Header)
|
removeHopByHopHeaders(response.Header)
|
||||||
|
@ -133,7 +133,7 @@ func HandleConnection(ctx context.Context, conn net.Conn, reader *std_bufio.Read
|
||||||
|
|
||||||
err = response.Write(conn)
|
err = response.Write(conn)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return common.AnyError(innerErr, err)
|
return E.Errors(innerErr, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if !keepAlive {
|
if !keepAlive {
|
||||||
|
|
|
@ -224,7 +224,7 @@ func HandleConnection0(ctx context.Context, conn net.Conn, version byte, authent
|
||||||
close(done)
|
close(done)
|
||||||
}()
|
}()
|
||||||
err = common.Error(io.Copy(io.Discard, conn))
|
err = common.Error(io.Copy(io.Discard, conn))
|
||||||
return common.AnyError(innerError, err)
|
return E.New(innerError, err)
|
||||||
default:
|
default:
|
||||||
err = socks5.WriteResponse(conn, socks5.Response{
|
err = socks5.WriteResponse(conn, socks5.Response{
|
||||||
ReplyCode: socks5.ReplyCodeUnsupported,
|
ReplyCode: socks5.ReplyCodeUnsupported,
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue