result.go 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. package result
  2. import (
  3. "fmt"
  4. "reflect"
  5. "runtime"
  6. mKh "gitp78su.ipnodns.ru/svi/kern/v4/lev0/helpers"
  7. )
  8. // Result — обёртка вокруг результата с возможной ошибкой.
  9. //
  10. // Может быть либо только полезное значение, либо только ошибка.
  11. type Result[T any] struct {
  12. val T // Полезное значение
  13. isErr bool // Если содержит ошибку
  14. isCheck bool // Если проверено
  15. err error // Ошибка
  16. }
  17. // NewRes -- возвращает успешный Result с значением.
  18. func NewRes[T any](result T) *Result[T] {
  19. // Для некоторых типов нужна дополнительная проверка через reflect
  20. v := reflect.ValueOf(result)
  21. switch v.Kind() {
  22. case reflect.Ptr, reflect.Slice, reflect.Map, reflect.Chan, reflect.Func, reflect.Interface:
  23. mKh.Hassert(!v.IsNil(), "NewOk(): result==nil")
  24. }
  25. sf := &Result[T]{
  26. val: result,
  27. }
  28. runtime.SetFinalizer(sf, sf.destroy)
  29. return sf
  30. }
  31. // NewErr -- возвращает Result с ошибкой.
  32. func NewErr[T any](err error) *Result[T] {
  33. mKh.Hassert(err != nil, "NewError(): err==nil")
  34. sf := &Result[T]{
  35. err: err,
  36. isErr: true,
  37. }
  38. runtime.SetFinalizer(sf, sf.destroy)
  39. return sf
  40. }
  41. // WrapErr -- оборачивает существующий Result с ошибкой с новой ошибкой.
  42. func WrapErr[T any](res *Result[T], err error) *Result[T] {
  43. mKh.Hassert(res.IsErr(), "WrapErr(): result not have error")
  44. mKh.Hassert(err != nil, "WrapErr(): err==nil")
  45. err0 := res.Err() // Чтобы не было паники при финализации и обернуть ошибку
  46. err = fmt.Errorf("%v, err=\n\t%w", err0, err)
  47. sf := NewErr[T](err)
  48. return sf
  49. }
  50. func (sf *Result[T]) destroy(res *Result[T]) {
  51. if !res.isCheck {
  52. mKh.Hassert(false, "Result[T].destroy(): err not checked")
  53. }
  54. }
  55. // IsOk -- возвращает true, если Result содержит значение.
  56. func (sf *Result[T]) IsOk() bool {
  57. sf.isCheck = true
  58. return !sf.isErr
  59. }
  60. // IsErr -- возвращает true, если Result содержит ошибку.
  61. func (sf *Result[T]) IsErr() bool {
  62. sf.isCheck = true
  63. return sf.isErr
  64. }
  65. // Val -- возвращает значение, если оно есть, иначе паникует.
  66. func (sf *Result[T]) Val() T {
  67. mKh.Hassert(!sf.isErr, "Result[T].Val(): err(%v)!=nil", sf.err)
  68. return sf.val
  69. }
  70. // ValOr -- возвращает значение, если оно есть, или значение по умолчанию.
  71. func (sf *Result[T]) ValOr(defaultValue T) T {
  72. sf.isCheck = true
  73. if sf.isErr {
  74. return defaultValue
  75. }
  76. return sf.val
  77. }
  78. // ValOrFn -- возвращает значение, если оно есть, или результат выполнения функции.
  79. func (sf *Result[T]) ValOrFn(fn func() T) T {
  80. mKh.Hassert(fn != nil, "Result[T].ValOrFn(): fn==nil")
  81. sf.isCheck = true
  82. if sf.isErr {
  83. return fn()
  84. }
  85. return sf.val
  86. }
  87. // Err -- возвращает ошибку, если она есть.
  88. func (sf *Result[T]) Err() error {
  89. mKh.Hassert(sf.isErr, "Result[T].Err(): err==nil")
  90. sf.isCheck = true
  91. return sf.err
  92. }
  93. // Error -- возвращает строковое представление ошибки, если она есть.
  94. func (sf *Result[T]) Error() string {
  95. mKh.Hassert(sf.isErr, "Result[T].Error(): err==nil")
  96. sf.isCheck = true
  97. return sf.err.Error()
  98. }
  99. // Hassert -- проверяет, что нет ошибки (с паникой).
  100. func (sf *Result[T]) Hassert(msgFormat string, args ...any) T {
  101. strErr := ""
  102. if sf.isErr {
  103. strErr = fmt.Sprintf(", err=\n\t%v", sf.err)
  104. }
  105. msg := fmt.Sprintf(msgFormat+strErr, args...)
  106. mKh.Hassert(!sf.isErr, msg)
  107. sf.isCheck = true
  108. return sf.val
  109. }
  110. // Assert -- проверяет, что нет ошибки (с паникой только на локальном стенде).
  111. func (sf *Result[T]) Assert(msgFormat string, args ...any) T {
  112. strErr := ""
  113. if sf.isErr {
  114. strErr = fmt.Sprintf(", err=\n\t%v", sf.err)
  115. }
  116. msg := fmt.Sprintf(msgFormat+strErr, args...)
  117. mKh.Assert(!sf.isErr, msg)
  118. sf.isCheck = true
  119. return sf.val
  120. }