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

basic_undo_log_builder.go 8.1 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307
  1. /*
  2. * Licensed to the Apache Software Foundation (ASF) under one or more
  3. * contributor license agreements. See the NOTICE file distributed with
  4. * this work for additional information regarding copyright ownership.
  5. * The ASF licenses this file to You under the Apache License, Version 2.0
  6. * (the "License"); you may not use this file except in compliance with
  7. * the License. You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. */
  17. package builder
  18. import (
  19. "bytes"
  20. "database/sql"
  21. "database/sql/driver"
  22. "fmt"
  23. "io"
  24. "strings"
  25. "github.com/arana-db/parser/ast"
  26. "github.com/arana-db/parser/test_driver"
  27. gxsort "github.com/dubbogo/gost/sort"
  28. "github.com/seata/seata-go/pkg/datasource/sql/types"
  29. )
  30. // todo the executor should be stateful
  31. type BasicUndoLogBuilder struct{}
  32. // GetScanSlice get the column type for scann
  33. // todo to use ColumnInfo get slice
  34. func (*BasicUndoLogBuilder) GetScanSlice(columnNames []string, tableMeta *types.TableMeta) []driver.Value {
  35. scanSlice := make([]driver.Value, 0, len(columnNames))
  36. for _, columnNmae := range columnNames {
  37. var (
  38. scanVal interface{}
  39. // 从metData获取该列的元信息
  40. columnMeta = tableMeta.Columns[columnNmae]
  41. )
  42. switch strings.ToUpper(columnMeta.DatabaseTypeString) {
  43. case "VARCHAR", "NVARCHAR", "VARCHAR2", "CHAR", "TEXT", "JSON", "TINYTEXT":
  44. scanVal = sql.RawBytes{}
  45. case "BIT", "INT", "LONGBLOB", "SMALLINT", "TINYINT", "BIGINT", "MEDIUMINT":
  46. if columnMeta.IsNullable == 0 {
  47. scanVal = int64(0)
  48. } else {
  49. scanVal = sql.NullInt64{}
  50. }
  51. case "DATE", "DATETIME", "TIME", "TIMESTAMP", "YEAR":
  52. scanVal = sql.NullTime{}
  53. case "DECIMAL", "DOUBLE", "FLOAT":
  54. if columnMeta.IsNullable == 0 {
  55. scanVal = float64(0)
  56. } else {
  57. scanVal = sql.NullFloat64{}
  58. }
  59. default:
  60. scanVal = sql.RawBytes{}
  61. }
  62. scanSlice = append(scanSlice, &scanVal)
  63. }
  64. return scanSlice
  65. }
  66. func (b *BasicUndoLogBuilder) buildSelectArgs(stmt *ast.SelectStmt, args []driver.Value) []driver.Value {
  67. var (
  68. selectArgsIndexs = make([]int32, 0)
  69. selectArgs = make([]driver.Value, 0)
  70. )
  71. b.traversalArgs(stmt.Where, &selectArgsIndexs)
  72. if stmt.OrderBy != nil {
  73. for _, item := range stmt.OrderBy.Items {
  74. b.traversalArgs(item, &selectArgsIndexs)
  75. }
  76. }
  77. if stmt.Limit != nil {
  78. if stmt.Limit.Offset != nil {
  79. b.traversalArgs(stmt.Limit.Offset, &selectArgsIndexs)
  80. }
  81. if stmt.Limit.Count != nil {
  82. b.traversalArgs(stmt.Limit.Count, &selectArgsIndexs)
  83. }
  84. }
  85. // sort selectArgs index array
  86. gxsort.Int32(selectArgsIndexs)
  87. for _, index := range selectArgsIndexs {
  88. selectArgs = append(selectArgs, args[index])
  89. }
  90. return selectArgs
  91. }
  92. // todo perfect all sql operation
  93. func (b *BasicUndoLogBuilder) traversalArgs(node ast.Node, argsIndex *[]int32) {
  94. if node == nil {
  95. return
  96. }
  97. switch node.(type) {
  98. case *ast.BinaryOperationExpr:
  99. expr := node.(*ast.BinaryOperationExpr)
  100. b.traversalArgs(expr.L, argsIndex)
  101. b.traversalArgs(expr.R, argsIndex)
  102. break
  103. case *ast.BetweenExpr:
  104. expr := node.(*ast.BetweenExpr)
  105. b.traversalArgs(expr.Left, argsIndex)
  106. b.traversalArgs(expr.Right, argsIndex)
  107. break
  108. case *ast.PatternInExpr:
  109. exprs := node.(*ast.PatternInExpr).List
  110. for i := 0; i < len(exprs); i++ {
  111. b.traversalArgs(exprs[i], argsIndex)
  112. }
  113. break
  114. case *test_driver.ParamMarkerExpr:
  115. *argsIndex = append(*argsIndex, int32(node.(*test_driver.ParamMarkerExpr).Order))
  116. break
  117. }
  118. }
  119. func (b *BasicUndoLogBuilder) buildRecordImages(rowsi driver.Rows, tableMetaData *types.TableMeta) (*types.RecordImage, error) {
  120. // select column names
  121. columnNames := rowsi.Columns()
  122. rowImages := make([]types.RowImage, 0)
  123. ss := b.GetScanSlice(columnNames, tableMetaData)
  124. for {
  125. err := rowsi.Next(ss)
  126. if err == io.EOF {
  127. break
  128. }
  129. if err != nil {
  130. return nil, err
  131. }
  132. columns := make([]types.ColumnImage, 0)
  133. // build record image
  134. for i, name := range columnNames {
  135. columnMeta := tableMetaData.Columns[name]
  136. keyType := types.IndexTypeNull
  137. if _, ok := tableMetaData.GetPrimaryKeyMap()[name]; ok {
  138. keyType = types.IndexTypePrimaryKey
  139. }
  140. jdbcType := types.MySQLStrToJavaType(columnMeta.DatabaseTypeString)
  141. columns = append(columns, types.ColumnImage{
  142. KeyType: keyType,
  143. ColumnName: name,
  144. ColumnType: jdbcType,
  145. Value: ss[i],
  146. })
  147. }
  148. rowImages = append(rowImages, types.RowImage{Columns: columns})
  149. }
  150. return &types.RecordImage{TableName: tableMetaData.TableName, Rows: rowImages}, nil
  151. }
  152. // buildWhereConditionByPKs build where condition by primary keys
  153. // each pk is a condition.the result will like :" (id,userCode) in ((?,?),(?,?)) or (id,userCode) in ((?,?),(?,?) ) or (id,userCode) in ((?,?))"
  154. func (b *BasicUndoLogBuilder) buildWhereConditionByPKs(pkNameList []string, rowSize int, dbType string, maxInSize int) string {
  155. var (
  156. whereStr = &strings.Builder{}
  157. batchSize = rowSize/maxInSize + 1
  158. )
  159. if rowSize%maxInSize == 0 {
  160. batchSize = rowSize / maxInSize
  161. }
  162. for batch := 0; batch < batchSize; batch++ {
  163. if batch > 0 {
  164. whereStr.WriteString(" OR ")
  165. }
  166. whereStr.WriteString("(")
  167. for i := 0; i < len(pkNameList); i++ {
  168. if i > 0 {
  169. whereStr.WriteString(",")
  170. }
  171. // todo add escape
  172. whereStr.WriteString(fmt.Sprintf("`%s`", pkNameList[i]))
  173. }
  174. whereStr.WriteString(") IN (")
  175. var eachSize int
  176. if batch == batchSize-1 {
  177. if rowSize%maxInSize == 0 {
  178. eachSize = maxInSize
  179. } else {
  180. eachSize = rowSize % maxInSize
  181. }
  182. } else {
  183. eachSize = maxInSize
  184. }
  185. for i := 0; i < eachSize; i++ {
  186. if i > 0 {
  187. whereStr.WriteString(",")
  188. }
  189. whereStr.WriteString("(")
  190. for j := 0; j < len(pkNameList); j++ {
  191. if j > 0 {
  192. whereStr.WriteString(",")
  193. }
  194. whereStr.WriteString("?")
  195. }
  196. whereStr.WriteString(")")
  197. }
  198. whereStr.WriteString(")")
  199. }
  200. return whereStr.String()
  201. }
  202. func (b *BasicUndoLogBuilder) buildPKParams(rows []types.RowImage, pkNameList []string) []driver.Value {
  203. params := make([]driver.Value, 0)
  204. for _, row := range rows {
  205. coumnMap := row.GetColumnMap()
  206. for _, pk := range pkNameList {
  207. col := coumnMap[pk]
  208. if col != nil {
  209. params = append(params, col.Value)
  210. }
  211. }
  212. }
  213. return params
  214. }
  215. // the string as local key. the local key example(multi pk): "t_user:1_a,2_b"
  216. func (b *BasicUndoLogBuilder) buildLockKey(rows driver.Rows, meta types.TableMeta) string {
  217. var (
  218. lockKeys bytes.Buffer
  219. filedSequence int
  220. )
  221. lockKeys.WriteString(meta.TableName)
  222. lockKeys.WriteString(":")
  223. pks := b.GetScanSlice(meta.GetPrimaryKeyOnlyName(), &meta)
  224. for {
  225. err := rows.Next(pks)
  226. if err == io.EOF {
  227. break
  228. }
  229. if filedSequence > 0 {
  230. lockKeys.WriteString(",")
  231. }
  232. pkSplitIndex := 0
  233. for _, value := range pks {
  234. if pkSplitIndex > 0 {
  235. lockKeys.WriteString("_")
  236. }
  237. lockKeys.WriteString(fmt.Sprintf("%v", value))
  238. pkSplitIndex++
  239. }
  240. filedSequence++
  241. }
  242. return lockKeys.String()
  243. }
  244. // the string as local key. the local key example(multi pk): "t_user:1_a,2_b"
  245. func (b *BasicUndoLogBuilder) buildLockKey2(records *types.RecordImage, meta types.TableMeta) string {
  246. var (
  247. lockKeys bytes.Buffer
  248. filedSequence int
  249. )
  250. lockKeys.WriteString(meta.TableName)
  251. lockKeys.WriteString(":")
  252. keys := meta.GetPrimaryKeyOnlyName()
  253. for _, row := range records.Rows {
  254. if filedSequence > 0 {
  255. lockKeys.WriteString(",")
  256. }
  257. pkSplitIndex := 0
  258. for _, column := range row.Columns {
  259. var hasKeyColumn bool
  260. for _, key := range keys {
  261. if column.ColumnName == key {
  262. hasKeyColumn = true
  263. if pkSplitIndex > 0 {
  264. lockKeys.WriteString("_")
  265. }
  266. lockKeys.WriteString(fmt.Sprintf("%v", column.Value))
  267. pkSplitIndex++
  268. }
  269. }
  270. if hasKeyColumn {
  271. filedSequence++
  272. }
  273. }
  274. }
  275. return lockKeys.String()
  276. }