| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190 |
- // Package errors is a drop-in replacement for Golang lib 'errors'.
- package errors // import "v2ray.com/core/common/errors"
- import (
- "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)
- }
|