errors.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  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. // String returns the string representation of this error.
  82. func (v *Error) String() string {
  83. return v.Error()
  84. }
  85. // WriteToLog writes current error into log.
  86. func (v *Error) WriteToLog() {
  87. log.Record(&log.GeneralMessage{
  88. Severity: GetSeverity(v),
  89. Content: v,
  90. })
  91. }
  92. // New returns a new error object with message formed from given arguments.
  93. func New(msg ...interface{}) *Error {
  94. return &Error{
  95. message: msg,
  96. severity: log.Severity_Info,
  97. }
  98. }
  99. // Cause returns the root cause of this error.
  100. func Cause(err error) error {
  101. if err == nil {
  102. return nil
  103. }
  104. for {
  105. inner, ok := err.(hasInnerError)
  106. if !ok || inner.Inner() == nil {
  107. break
  108. }
  109. err = inner.Inner()
  110. }
  111. return err
  112. }
  113. // GetSeverity returns the actual severity of the error, including inner errors.
  114. func GetSeverity(err error) log.Severity {
  115. if s, ok := err.(hasSeverity); ok {
  116. return s.Severity()
  117. }
  118. return log.Severity_Info
  119. }