errors.go 2.2 KB

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