parse_result.go 5.9 KB

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