| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180 | // Package errors is a drop-in replacement for Golang lib 'errors'.package errors // import "v2ray.com/core/common/errors"import (	"os"	"strings"	"v2ray.com/core/common/log"	"v2ray.com/core/common/serial")type hasInnerError interface {	// Inner returns the underlying error of this one.	Inner() error}type hasSeverity interface {	Severity() log.Severity}// Error is an error object with underlying error.type Error struct {	prefix   []interface{}	path     []string	message  []interface{}	inner    error	severity log.Severity}// Error implements error.Error().func (v *Error) Error() string {	msg := serial.Concat(v.message...)	if v.inner != nil {		msg += " > " + v.inner.Error()	}	if len(v.path) > 0 {		msg = strings.Join(v.path, "|") + ": " + msg	}	var prefix string	for _, p := range v.prefix {		prefix += "[" + serial.ToString(p) + "] "	}	return prefix + msg}// Inner implements hasInnerError.Inner()func (v *Error) Inner() error {	if v.inner == nil {		return nil	}	return v.inner}func (v *Error) Base(err error) *Error {	v.inner = err	return v}func (v *Error) atSeverity(s log.Severity) *Error {	v.severity = s	return v}func (v *Error) Severity() log.Severity {	if v.inner == nil {		return v.severity	}	if s, ok := v.inner.(hasSeverity); ok {		as := s.Severity()		if as < v.severity {			return as		}	}	return v.severity}// AtDebug sets the severity to debug.func (v *Error) AtDebug() *Error {	return v.atSeverity(log.Severity_Debug)}// AtInfo sets the severity to info.func (v *Error) AtInfo() *Error {	return v.atSeverity(log.Severity_Info)}// AtWarning sets the severity to warning.func (v *Error) AtWarning() *Error {	return v.atSeverity(log.Severity_Warning)}// AtError sets the severity to error.func (v *Error) AtError() *Error {	return v.atSeverity(log.Severity_Error)}// Path sets the path to the location where this error happens.func (v *Error) Path(path ...string) *Error {	v.path = path	return v}// String returns the string representation of this error.func (v *Error) String() string {	return v.Error()}// WriteToLog writes current error into log.func (v *Error) WriteToLog(opts ...ExportOption) {	var holder ExportOptionHolder	for _, opt := range opts {		opt(&holder)	}	if holder.SessionID > 0 {		v.prefix = append(v.prefix, holder.SessionID)	}	log.Record(&log.GeneralMessage{		Severity: GetSeverity(v),		Content:  v,	})}type ExportOptionHolder struct {	SessionID uint32}type ExportOption func(*ExportOptionHolder)// New returns a new error object with message formed from given arguments.func New(msg ...interface{}) *Error {	return &Error{		message:  msg,		severity: log.Severity_Info,	}}// Cause returns the root cause of this error.func Cause(err error) error {	if err == nil {		return nil	}L:	for {		switch inner := err.(type) {		case hasInnerError:			if inner.Inner() == nil {				break L			}			err = inner.Inner()		case *os.PathError:			if inner.Err == nil {				break L			}			err = inner.Err		case *os.SyscallError:			if inner.Err == nil {				break L			}			err = inner.Err		default:			break L		}	}	return err}// GetSeverity returns the actual severity of the error, including inner errors.func GetSeverity(err error) log.Severity {	if s, ok := err.(hasSeverity); ok {		return s.Severity()	}	return log.Severity_Info}
 |