errors.go 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. // Package errors is a drop-in replacement for Golang lib 'errors'.
  2. package errors
  3. import (
  4. "strings"
  5. "v2ray.com/core/common/log"
  6. "v2ray.com/core/common/serial"
  7. )
  8. type hasInnerError interface {
  9. // Inner returns the underlying error of this one.
  10. Inner() error
  11. }
  12. type hasSeverity interface {
  13. Severity() log.Severity
  14. }
  15. // Error is an error object with underlying error.
  16. type Error struct {
  17. message []interface{}
  18. inner error
  19. severity log.Severity
  20. path []string
  21. }
  22. // Error implements error.Error().
  23. func (v *Error) Error() string {
  24. msg := serial.Concat(v.message...)
  25. if v.inner != nil {
  26. msg += " > " + v.inner.Error()
  27. }
  28. if len(v.path) > 0 {
  29. msg = strings.Join(v.path, "|") + ": " + msg
  30. }
  31. return msg
  32. }
  33. // Inner implements hasInnerError.Inner()
  34. func (v *Error) Inner() error {
  35. if v.inner == nil {
  36. return nil
  37. }
  38. return v.inner
  39. }
  40. func (v *Error) Base(err error) *Error {
  41. v.inner = err
  42. return v
  43. }
  44. func (v *Error) atSeverity(s log.Severity) *Error {
  45. v.severity = s
  46. return v
  47. }
  48. func (v *Error) Severity() log.Severity {
  49. if v.inner == nil {
  50. return v.severity
  51. }
  52. if s, ok := v.inner.(hasSeverity); ok {
  53. as := s.Severity()
  54. if as < v.severity {
  55. return as
  56. }
  57. }
  58. return v.severity
  59. }
  60. // AtDebug sets the severity to debug.
  61. func (v *Error) AtDebug() *Error {
  62. return v.atSeverity(log.Severity_Debug)
  63. }
  64. // AtInfo sets the severity to info.
  65. func (v *Error) AtInfo() *Error {
  66. return v.atSeverity(log.Severity_Info)
  67. }
  68. // AtWarning sets the severity to warning.
  69. func (v *Error) AtWarning() *Error {
  70. return v.atSeverity(log.Severity_Warning)
  71. }
  72. // AtError sets the severity to error.
  73. func (v *Error) AtError() *Error {
  74. return v.atSeverity(log.Severity_Error)
  75. }
  76. // Path sets the path to the location where this error happens.
  77. func (v *Error) Path(path ...string) *Error {
  78. v.path = path
  79. return v
  80. }
  81. // New returns a new error object with message formed from given arguments.
  82. func New(msg ...interface{}) *Error {
  83. return &Error{
  84. message: msg,
  85. severity: log.Severity_Info,
  86. }
  87. }
  88. // Cause returns the root cause of this error.
  89. func Cause(err error) error {
  90. if err == nil {
  91. return nil
  92. }
  93. for {
  94. inner, ok := err.(hasInnerError)
  95. if !ok || inner.Inner() == nil {
  96. break
  97. }
  98. err = inner.Inner()
  99. }
  100. return err
  101. }
  102. func GetSeverity(err error) log.Severity {
  103. if s, ok := err.(hasSeverity); ok {
  104. return s.Severity()
  105. }
  106. return log.Severity_Info
  107. }