| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190 | // Package errors is a drop-in replacement for Golang lib 'errors'.package errorsimport (	"context"	"strings"	"v2ray.com/core/common/log"	"v2ray.com/core/common/serial"	"v2ray.com/core/common/session")type hasInnerError interface {	// Inner returns the underlying error of this one.	Inner() error}type hasSeverity interface {	Severity() log.Severity}type hasContext interface {	Context() context.Context}// Error is an error object with underlying error.type Error struct {	message  []interface{}	inner    error	severity log.Severity	path     []string	ctx      context.Context}// 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	}	return 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) WithContext(ctx context.Context) *Error {	v.ctx = ctx	return v}func (v *Error) Context() context.Context {	if v.ctx != nil {		return v.ctx	}	if v.inner == nil {		return nil	}	if c, ok := v.inner.(hasContext); ok {		return c.Context()	}	return nil}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() {	ctx := v.Context()	var sid session.ID	if ctx != nil {		sid = session.IDFromContext(ctx)	}	var c interface{} = v	if sid > 0 {		c = sessionLog{			id:      sid,			content: v,		}	}	log.Record(&log.GeneralMessage{		Severity: GetSeverity(v),		Content:  c,	})}// 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	}	for {		inner, ok := err.(hasInnerError)		if !ok || inner.Inner() == nil {			break		}		err = inner.Inner()	}	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}type sessionLog struct {	id      session.ID	content interface{}}func (s sessionLog) String() string {	return serial.Concat("[", s.id, "] ", s.content)}
 |