parse_result.go 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. package sql
  2. import (
  3. "errors"
  4. "fmt"
  5. "git.sxidc.com/go-tools/utils/encoding"
  6. "git.sxidc.com/go-tools/utils/reflectutils"
  7. "git.sxidc.com/go-tools/utils/strutils"
  8. "git.sxidc.com/service-supports/ds-sdk/sdk"
  9. "reflect"
  10. "strings"
  11. "time"
  12. )
  13. func ParseSqlResult[T any](input any) (T, error) {
  14. var zero T
  15. if input == nil {
  16. return zero, nil
  17. }
  18. // 构造需要遍历的tableRows
  19. tableRows, ok := input.([]sdk.SqlResult)
  20. if !ok {
  21. tableRow, ok := input.(sdk.SqlResult)
  22. if !ok {
  23. return zero, errors.New("输入数据应该为sdk.SqlResult或[]sdk.SqlResult")
  24. }
  25. tableRows = []sdk.SqlResult{tableRow}
  26. }
  27. // 构造outputValue
  28. typeCheckErr := errors.New("可以接受的类型为struct, *struct, []struct, []*struct")
  29. outputType := reflect.TypeOf(zero)
  30. if outputType.Kind() != reflect.Struct && outputType.Kind() != reflect.Ptr && outputType.Kind() != reflect.Slice {
  31. return zero, typeCheckErr
  32. } else if outputType.Kind() == reflect.Ptr && outputType.Elem().Kind() != reflect.Struct {
  33. return zero, typeCheckErr
  34. } else if outputType.Kind() == reflect.Slice &&
  35. (outputType.Elem().Kind() != reflect.Struct && outputType.Elem().Kind() != reflect.Ptr) {
  36. return zero, typeCheckErr
  37. } else if outputType.Kind() == reflect.Slice &&
  38. outputType.Elem().Kind() == reflect.Ptr && outputType.Elem().Elem().Kind() != reflect.Struct {
  39. return zero, typeCheckErr
  40. }
  41. var outputValue reflect.Value
  42. if outputType.Kind() == reflect.Struct || outputType.Kind() == reflect.Ptr {
  43. outputValue = reflect.New(outputType).Elem()
  44. } else {
  45. outputValue = reflect.MakeSlice(outputType, 0, 0)
  46. }
  47. for _, tableRow := range tableRows {
  48. // 构造输出实体
  49. var outputEntity any
  50. if outputType.Kind() == reflect.Struct {
  51. outputEntity = outputValue.Addr().Interface()
  52. } else if outputType.Kind() == reflect.Ptr {
  53. outputEntity = reflect.New(outputType.Elem()).Interface()
  54. } else {
  55. if outputType.Elem().Kind() == reflect.Struct {
  56. outputEntity = reflect.New(outputType.Elem()).Interface()
  57. } else {
  58. outputValueElemPtr := reflect.New(outputType.Elem()).Elem()
  59. outputValueElemPtr.Set(reflect.New(outputType.Elem().Elem()))
  60. outputEntity = outputValueElemPtr.Interface()
  61. }
  62. }
  63. sqlResult, err := ParseSqlResultTag(outputEntity)
  64. if err != nil {
  65. return zero, err
  66. }
  67. err = formOutputEntity(tableRow, sqlResult)
  68. if err != nil {
  69. return zero, err
  70. }
  71. if outputType.Kind() == reflect.Ptr {
  72. outputValue = reflect.ValueOf(outputEntity)
  73. } else if outputType.Kind() == reflect.Slice {
  74. if outputType.Elem().Kind() == reflect.Struct {
  75. outputValue = reflect.Append(outputValue, reflect.ValueOf(outputEntity).Elem())
  76. } else {
  77. outputValue = reflect.Append(outputValue, reflect.ValueOf(outputEntity))
  78. }
  79. }
  80. }
  81. output, ok := outputValue.Interface().(T)
  82. if !ok {
  83. return zero, errors.New("输出类型不匹配")
  84. }
  85. return output, nil
  86. }
  87. func formOutputEntity(tableRow sdk.SqlResult, sqlResult *Result) error {
  88. for fieldName, resultElement := range sqlResult.ResultElement {
  89. switch element := resultElement.(type) {
  90. case *Result:
  91. err := formOutputEntity(tableRow, element)
  92. if err != nil {
  93. return err
  94. }
  95. case *ResultColumn:
  96. tableRowValue, ok := tableRow[element.Name]
  97. if !ok {
  98. continue
  99. }
  100. if tableRowValue == nil {
  101. continue
  102. }
  103. // 构造结构字段,如果结构字段是指针且为nil,需要构造元素
  104. fieldTypeElem := element.FieldTypeElem
  105. fieldValueElem := element.FieldValueElem
  106. outputKind := reflectutils.GroupValueKind(fieldValueElem)
  107. if !fieldValueElem.CanSet() {
  108. continue
  109. }
  110. switch outputKind {
  111. case reflect.Bool:
  112. err := reflectutils.AssignBoolValue(tableRowValue, fieldValueElem)
  113. if err != nil {
  114. return err
  115. }
  116. case reflect.String:
  117. strValue := tableRowValue.(string)
  118. if strutils.IsStringNotEmpty(element.ParseTime) {
  119. parsedTime, err := sdk.ParseSqlResultTimeStr(strValue)
  120. if err != nil {
  121. return err
  122. }
  123. strValue = parsedTime.Format(element.ParseTime)
  124. } else if strutils.IsStringNotEmpty(element.AESKey) {
  125. if strutils.IsStringNotEmpty(strValue) {
  126. decryptedValue, err := encoding.AESDecrypt(strValue, element.AESKey)
  127. if err != nil {
  128. return err
  129. }
  130. strValue = decryptedValue
  131. }
  132. }
  133. err := reflectutils.AssignStringValue(strValue, fieldValueElem)
  134. if err != nil {
  135. return err
  136. }
  137. case reflect.Int64:
  138. err := reflectutils.AssignInt64Value(tableRowValue, fieldValueElem)
  139. if err != nil {
  140. return err
  141. }
  142. case reflect.Uint64:
  143. err := reflectutils.AssignUint64Value(tableRowValue, fieldValueElem)
  144. if err != nil {
  145. return err
  146. }
  147. case reflect.Float64:
  148. err := reflectutils.AssignFloat64Value(tableRowValue, fieldValueElem)
  149. if err != nil {
  150. return err
  151. }
  152. case reflect.Struct:
  153. if fieldValueElem.Type() == reflect.TypeOf(time.Time{}) {
  154. parsedTime, err := sdk.ParseSqlResultTimeStr(tableRowValue.(string))
  155. if err != nil {
  156. return err
  157. }
  158. fieldValueElem.Set(reflect.ValueOf(parsedTime))
  159. continue
  160. }
  161. return fmt.Errorf("字段: %s 列: %s 不支持的类型: %s",
  162. fieldName, element.Name, fieldTypeElem.String())
  163. case reflect.Slice:
  164. if fieldTypeElem.Elem().Kind() != reflect.String {
  165. return errors.New("slice仅支持[]string")
  166. }
  167. strValue, ok := tableRowValue.(string)
  168. if !ok {
  169. return errors.New("slice仅支持[]string")
  170. }
  171. strParts := strings.Split(strValue, element.SplitWith)
  172. if strParts == nil || len(strParts) == 0 {
  173. return nil
  174. }
  175. valSlice := fieldValueElem
  176. if valSlice.IsNil() {
  177. valSlice = reflect.MakeSlice(fieldTypeElem, 0, 0)
  178. }
  179. for _, strPart := range strParts {
  180. valSlice = reflect.Append(valSlice, reflect.ValueOf(strPart))
  181. }
  182. fieldValueElem.Set(valSlice)
  183. default:
  184. return fmt.Errorf("字段: %s 列: %s 不支持的类型: %s",
  185. fieldName, element.Name, fieldTypeElem.String())
  186. }
  187. default:
  188. return errors.New("不支持的元素类型")
  189. }
  190. }
  191. return nil
  192. }