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.

conn_at_test.go 6.8 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  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 sql
  18. import (
  19. "context"
  20. "database/sql"
  21. "database/sql/driver"
  22. "sync/atomic"
  23. "testing"
  24. "github.com/golang/mock/gomock"
  25. "github.com/google/uuid"
  26. "github.com/stretchr/testify/assert"
  27. "seata.apache.org/seata-go/pkg/datasource/sql/exec"
  28. "seata.apache.org/seata-go/pkg/datasource/sql/mock"
  29. "seata.apache.org/seata-go/pkg/datasource/sql/types"
  30. "seata.apache.org/seata-go/pkg/protocol/branch"
  31. "seata.apache.org/seata-go/pkg/tm"
  32. )
  33. func TestMain(m *testing.M) {
  34. Init()
  35. m.Run()
  36. }
  37. func initAtConnTestResource(t *testing.T) (*gomock.Controller, *sql.DB, *mockSQLInterceptor, *mockTxHook) {
  38. ctrl := gomock.NewController(t)
  39. mockMgr := initMockResourceManager(branch.BranchTypeAT, ctrl)
  40. _ = mockMgr
  41. db, err := sql.Open(SeataATMySQLDriver, "root:12345678@tcp(127.0.0.1:3306)/seata_client?multiStatements=true")
  42. if err != nil {
  43. t.Fatal(err)
  44. }
  45. _ = initMockAtConnector(t, ctrl, db, func(t *testing.T, ctrl *gomock.Controller) driver.Connector {
  46. mockTx := mock.NewMockTestDriverTx(ctrl)
  47. mockTx.EXPECT().Commit().AnyTimes().Return(nil)
  48. mockTx.EXPECT().Rollback().AnyTimes().Return(nil)
  49. mockConn := mock.NewMockTestDriverConn(ctrl)
  50. mockConn.EXPECT().Begin().AnyTimes().Return(mockTx, nil)
  51. mockConn.EXPECT().BeginTx(gomock.Any(), gomock.Any()).AnyTimes().Return(mockTx, nil)
  52. mockConn.EXPECT().QueryContext(gomock.Any(), gomock.Any(), gomock.Any()).AnyTimes().DoAndReturn(
  53. func(ctx context.Context, query string, args []driver.NamedValue) (driver.Rows, error) {
  54. rows := &mysqlMockRows{}
  55. rows.data = [][]interface{}{
  56. {"8.0.29"},
  57. }
  58. return rows, nil
  59. })
  60. baseMockConn(mockConn)
  61. connector := mock.NewMockTestDriverConnector(ctrl)
  62. connector.EXPECT().Connect(gomock.Any()).AnyTimes().Return(mockConn, nil)
  63. return connector
  64. })
  65. mi := &mockSQLInterceptor{}
  66. ti := &mockTxHook{}
  67. exec.CleanCommonHook()
  68. CleanTxHooks()
  69. exec.RegisterCommonHook(mi)
  70. RegisterTxHook(ti)
  71. return ctrl, db, mi, ti
  72. }
  73. func TestATConn_ExecContext(t *testing.T) {
  74. ctrl, db, mi, ti := initAtConnTestResource(t)
  75. defer func() {
  76. ctrl.Finish()
  77. db.Close()
  78. CleanTxHooks()
  79. }()
  80. t.Run("have xid", func(t *testing.T) {
  81. ctx := tm.InitSeataContext(context.Background())
  82. tm.SetXID(ctx, uuid.New().String())
  83. t.Logf("set xid=%s", tm.GetXID(ctx))
  84. beforeHook := func(_ context.Context, execCtx *types.ExecContext) {
  85. t.Logf("on exec xid=%s", execCtx.TxCtx.XID)
  86. assert.Equal(t, tm.GetXID(ctx), execCtx.TxCtx.XID)
  87. assert.Equal(t, types.ATMode, execCtx.TxCtx.TransactionMode)
  88. }
  89. mi.before = beforeHook
  90. var comitCnt int32
  91. beforeCommit := func(tx *Tx) {
  92. atomic.AddInt32(&comitCnt, 1)
  93. assert.Equal(t, types.ATMode, tx.tranCtx.TransactionMode)
  94. }
  95. ti.beforeCommit = beforeCommit
  96. conn, err := db.Conn(context.Background())
  97. assert.NoError(t, err)
  98. _, err = conn.ExecContext(ctx, "SELECT 1")
  99. assert.NoError(t, err)
  100. _, err = db.ExecContext(ctx, "SELECT 1")
  101. assert.NoError(t, err)
  102. assert.Equal(t, int32(2), atomic.LoadInt32(&comitCnt))
  103. })
  104. t.Run("not xid", func(t *testing.T) {
  105. mi.before = func(_ context.Context, execCtx *types.ExecContext) {
  106. assert.Equal(t, "", execCtx.TxCtx.XID)
  107. assert.Equal(t, types.Local, execCtx.TxCtx.TransactionMode)
  108. }
  109. var comitCnt int32
  110. ti.beforeCommit = func(tx *Tx) {
  111. atomic.AddInt32(&comitCnt, 1)
  112. }
  113. conn, err := db.Conn(context.Background())
  114. assert.NoError(t, err)
  115. _, err = conn.ExecContext(context.Background(), "SELECT 1")
  116. assert.NoError(t, err)
  117. _, err = db.ExecContext(context.Background(), "SELECT 1")
  118. assert.NoError(t, err)
  119. _, err = db.Exec("SELECT 1")
  120. assert.NoError(t, err)
  121. assert.Equal(t, int32(0), atomic.LoadInt32(&comitCnt))
  122. })
  123. }
  124. func TestATConn_BeginTx(t *testing.T) {
  125. ctrl, db, mi, ti := initAtConnTestResource(t)
  126. defer func() {
  127. ctrl.Finish()
  128. db.Close()
  129. CleanTxHooks()
  130. }()
  131. t.Run("tx-local", func(t *testing.T) {
  132. tx, err := db.Begin()
  133. assert.NoError(t, err)
  134. mi.before = func(_ context.Context, execCtx *types.ExecContext) {
  135. assert.Equal(t, "", execCtx.TxCtx.XID)
  136. assert.Equal(t, types.Local, execCtx.TxCtx.TransactionMode)
  137. }
  138. var comitCnt int32
  139. ti.beforeCommit = func(tx *Tx) {
  140. atomic.AddInt32(&comitCnt, 1)
  141. }
  142. _, err = tx.ExecContext(context.Background(), "SELECT * FROM user")
  143. assert.NoError(t, err)
  144. _, err = tx.ExecContext(tm.InitSeataContext(context.Background()), "SELECT * FROM user")
  145. assert.NoError(t, err)
  146. err = tx.Commit()
  147. assert.NoError(t, err)
  148. assert.Equal(t, int32(1), atomic.LoadInt32(&comitCnt))
  149. })
  150. t.Run("tx-local-context", func(t *testing.T) {
  151. tx, err := db.BeginTx(context.Background(), &sql.TxOptions{})
  152. assert.NoError(t, err)
  153. mi.before = func(_ context.Context, execCtx *types.ExecContext) {
  154. assert.Equal(t, "", execCtx.TxCtx.XID)
  155. assert.Equal(t, types.Local, execCtx.TxCtx.TransactionMode)
  156. }
  157. var comitCnt int32
  158. ti.beforeCommit = func(tx *Tx) {
  159. atomic.AddInt32(&comitCnt, 1)
  160. }
  161. _, err = tx.ExecContext(context.Background(), "SELECT * FROM user")
  162. assert.NoError(t, err)
  163. _, err = tx.ExecContext(tm.InitSeataContext(context.Background()), "SELECT * FROM user")
  164. assert.NoError(t, err)
  165. err = tx.Commit()
  166. assert.NoError(t, err)
  167. assert.Equal(t, int32(1), atomic.LoadInt32(&comitCnt))
  168. })
  169. t.Run("tx-at-context", func(t *testing.T) {
  170. ctx := tm.InitSeataContext(context.Background())
  171. tm.SetXID(ctx, uuid.NewString())
  172. tx, err := db.BeginTx(ctx, &sql.TxOptions{})
  173. assert.NoError(t, err)
  174. mi.before = func(_ context.Context, execCtx *types.ExecContext) {
  175. assert.Equal(t, tm.GetXID(ctx), execCtx.TxCtx.XID)
  176. assert.Equal(t, types.ATMode, execCtx.TxCtx.TransactionMode)
  177. }
  178. var comitCnt int32
  179. ti.beforeCommit = func(tx *Tx) {
  180. atomic.AddInt32(&comitCnt, 1)
  181. }
  182. _, err = tx.ExecContext(context.Background(), "SELECT * FROM user")
  183. assert.NoError(t, err)
  184. _, err = tx.ExecContext(context.Background(), "SELECT * FROM user")
  185. assert.NoError(t, err)
  186. err = tx.Commit()
  187. assert.NoError(t, err)
  188. assert.Equal(t, int32(1), atomic.LoadInt32(&comitCnt))
  189. })
  190. }