You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

173 lines
3.3 KiB

  1. package proto
  2. import (
  3. "encoding"
  4. "fmt"
  5. "reflect"
  6. "time"
  7. "github.com/go-redis/redis/v8/internal/util"
  8. )
  9. // Scan parses bytes `b` to `v` with appropriate type.
  10. func Scan(b []byte, v interface{}) error {
  11. switch v := v.(type) {
  12. case nil:
  13. return fmt.Errorf("redis: Scan(nil)")
  14. case *string:
  15. *v = util.BytesToString(b)
  16. return nil
  17. case *[]byte:
  18. *v = b
  19. return nil
  20. case *int:
  21. var err error
  22. *v, err = util.Atoi(b)
  23. return err
  24. case *int8:
  25. n, err := util.ParseInt(b, 10, 8)
  26. if err != nil {
  27. return err
  28. }
  29. *v = int8(n)
  30. return nil
  31. case *int16:
  32. n, err := util.ParseInt(b, 10, 16)
  33. if err != nil {
  34. return err
  35. }
  36. *v = int16(n)
  37. return nil
  38. case *int32:
  39. n, err := util.ParseInt(b, 10, 32)
  40. if err != nil {
  41. return err
  42. }
  43. *v = int32(n)
  44. return nil
  45. case *int64:
  46. n, err := util.ParseInt(b, 10, 64)
  47. if err != nil {
  48. return err
  49. }
  50. *v = n
  51. return nil
  52. case *uint:
  53. n, err := util.ParseUint(b, 10, 64)
  54. if err != nil {
  55. return err
  56. }
  57. *v = uint(n)
  58. return nil
  59. case *uint8:
  60. n, err := util.ParseUint(b, 10, 8)
  61. if err != nil {
  62. return err
  63. }
  64. *v = uint8(n)
  65. return nil
  66. case *uint16:
  67. n, err := util.ParseUint(b, 10, 16)
  68. if err != nil {
  69. return err
  70. }
  71. *v = uint16(n)
  72. return nil
  73. case *uint32:
  74. n, err := util.ParseUint(b, 10, 32)
  75. if err != nil {
  76. return err
  77. }
  78. *v = uint32(n)
  79. return nil
  80. case *uint64:
  81. n, err := util.ParseUint(b, 10, 64)
  82. if err != nil {
  83. return err
  84. }
  85. *v = n
  86. return nil
  87. case *float32:
  88. n, err := util.ParseFloat(b, 32)
  89. if err != nil {
  90. return err
  91. }
  92. *v = float32(n)
  93. return err
  94. case *float64:
  95. var err error
  96. *v, err = util.ParseFloat(b, 64)
  97. return err
  98. case *bool:
  99. *v = len(b) == 1 && b[0] == '1'
  100. return nil
  101. case *time.Time:
  102. var err error
  103. *v, err = time.Parse(time.RFC3339Nano, util.BytesToString(b))
  104. return err
  105. case encoding.BinaryUnmarshaler:
  106. return v.UnmarshalBinary(b)
  107. default:
  108. return fmt.Errorf(
  109. "redis: can't unmarshal %T (consider implementing BinaryUnmarshaler)", v)
  110. }
  111. }
  112. func ScanSlice(data []string, slice interface{}) error {
  113. v := reflect.ValueOf(slice)
  114. if !v.IsValid() {
  115. return fmt.Errorf("redis: ScanSlice(nil)")
  116. }
  117. if v.Kind() != reflect.Ptr {
  118. return fmt.Errorf("redis: ScanSlice(non-pointer %T)", slice)
  119. }
  120. v = v.Elem()
  121. if v.Kind() != reflect.Slice {
  122. return fmt.Errorf("redis: ScanSlice(non-slice %T)", slice)
  123. }
  124. next := makeSliceNextElemFunc(v)
  125. for i, s := range data {
  126. elem := next()
  127. if err := Scan([]byte(s), elem.Addr().Interface()); err != nil {
  128. err = fmt.Errorf("redis: ScanSlice index=%d value=%q failed: %w", i, s, err)
  129. return err
  130. }
  131. }
  132. return nil
  133. }
  134. func makeSliceNextElemFunc(v reflect.Value) func() reflect.Value {
  135. elemType := v.Type().Elem()
  136. if elemType.Kind() == reflect.Ptr {
  137. elemType = elemType.Elem()
  138. return func() reflect.Value {
  139. if v.Len() < v.Cap() {
  140. v.Set(v.Slice(0, v.Len()+1))
  141. elem := v.Index(v.Len() - 1)
  142. if elem.IsNil() {
  143. elem.Set(reflect.New(elemType))
  144. }
  145. return elem.Elem()
  146. }
  147. elem := reflect.New(elemType)
  148. v.Set(reflect.Append(v, elem))
  149. return elem.Elem()
  150. }
  151. }
  152. zero := reflect.Zero(elemType)
  153. return func() reflect.Value {
  154. if v.Len() < v.Cap() {
  155. v.Set(v.Slice(0, v.Len()+1))
  156. return v.Index(v.Len() - 1)
  157. }
  158. v.Set(reflect.Append(v, zero))
  159. return v.Index(v.Len() - 1)
  160. }
  161. }