parse_table_row.go 5.9 KB

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