kresult.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. package ktypes
  2. import (
  3. "fmt"
  4. "reflect"
  5. "runtime"
  6. . "gitp78su.ipnodns.ru/svi/kern/kc/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. // NewOk -- возвращает успешный Result с значением
  18. func NewOk[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. 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. 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. Hassert(res.IsErr(), "WrapErr(): result not have error")
  44. 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. runtime.SetFinalizer(&sf, sf.destroy)
  49. return sf
  50. }
  51. func (sf *Result[T]) destroy(res *Result[T]) {
  52. if !res.isCheck {
  53. Hassert(false, "Result[T].destroy(): err not checked")
  54. }
  55. }
  56. // IsOk -- возвращает true, если Result содержит значение
  57. func (sf *Result[T]) IsOk() bool {
  58. sf.isCheck = true
  59. return !sf.isErr
  60. }
  61. // IsErr -- возвращает true, если Result содержит ошибку
  62. func (sf *Result[T]) IsErr() bool {
  63. sf.isCheck = true
  64. return sf.isErr
  65. }
  66. // Val -- возвращает значение, если оно есть, иначе паникует
  67. func (sf *Result[T]) Val() T {
  68. Hassert(!sf.isErr, "Result[T].Val(): err(%v)!=nil", sf.err)
  69. return sf.val
  70. }
  71. // ValOr -- возвращает значение, если оно есть, или значение по умолчанию
  72. func (sf *Result[T]) ValOr(defaultValue T) T {
  73. sf.isCheck = true
  74. if sf.isErr {
  75. return defaultValue
  76. }
  77. return sf.val
  78. }
  79. // ValOrFn -- возвращает значение, если оно есть, или результат выполнения функции
  80. func (sf *Result[T]) ValOrFn(fn func() T) T {
  81. Hassert(fn != nil, "Result[T].ValOrFn(): fn==nil")
  82. sf.isCheck = true
  83. if sf.isErr {
  84. return fn()
  85. }
  86. return sf.val
  87. }
  88. // Err -- возвращает ошибку, если она есть
  89. func (sf *Result[T]) Err() error {
  90. Hassert(sf.isErr, "Result[T].Err(): err==nil")
  91. sf.isCheck = true
  92. return sf.err
  93. }
  94. // Error -- возвращает строковое представление ошибки, если она есть
  95. func (sf *Result[T]) Error() string {
  96. Hassert(sf.isErr, "Result[T].Error(): err==nil")
  97. sf.isCheck = true
  98. return sf.err.Error()
  99. }
  100. // Hassert -- проверяет, что нет ошибки (с паникой)
  101. func (sf *Result[T]) Hassert(msgFormat string, args ...any) {
  102. strErr := ""
  103. if sf.isErr {
  104. strErr = fmt.Sprintf(", err=\n\t%v", sf.err)
  105. }
  106. msg := fmt.Sprintf(msgFormat+strErr, args...)
  107. Hassert(!sf.isErr, msg)
  108. sf.isCheck = true
  109. }
  110. // Assert -- проверяет, что нет ошибки (с паникой только на локальном стенде)
  111. func (sf *Result[T]) Assert(msgFormat string, args ...any) {
  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. Assert(!sf.isErr, msg)
  118. sf.isCheck = true
  119. }