/* * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You under the Apache License, Version 2.0 * (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package sql import ( "context" "database/sql" "database/sql/driver" "io" "sync/atomic" "testing" "time" "github.com/bluele/gcache" "github.com/golang/mock/gomock" "github.com/google/uuid" "github.com/stretchr/testify/assert" "seata.apache.org/seata-go/pkg/datasource/sql/exec" "seata.apache.org/seata-go/pkg/datasource/sql/mock" "seata.apache.org/seata-go/pkg/datasource/sql/types" "seata.apache.org/seata-go/pkg/protocol/branch" "seata.apache.org/seata-go/pkg/tm" ) type mysqlMockRows struct { idx int data [][]interface{} } func (m *mysqlMockRows) Columns() []string { //TODO implement me panic("implement me") } func (m *mysqlMockRows) Close() error { //TODO implement me panic("implement me") } func (m *mysqlMockRows) Next(dest []driver.Value) error { if m.idx == len(m.data) { return io.EOF } min := func(a, b int) int { if a < b { return a } return b } cnt := min(len(m.data[0]), len(dest)) for i := 0; i < cnt; i++ { dest[i] = m.data[m.idx][i] } m.idx++ return nil } type mockSQLInterceptor struct { before func(ctx context.Context, execCtx *types.ExecContext) after func(ctx context.Context, execCtx *types.ExecContext) } func (mi *mockSQLInterceptor) Type() types.SQLType { return types.SQLTypeUnknown } // Before func (mi *mockSQLInterceptor) Before(ctx context.Context, execCtx *types.ExecContext) error { if mi.before != nil { mi.before(ctx, execCtx) } return nil } // After func (mi *mockSQLInterceptor) After(ctx context.Context, execCtx *types.ExecContext) error { if mi.after != nil { mi.after(ctx, execCtx) } return nil } type mockTxHook struct { beforeCommit func(tx *Tx) beforeRollback func(tx *Tx) } // BeforeCommit func (mi *mockTxHook) BeforeCommit(tx *Tx) { if mi.beforeCommit != nil { mi.beforeCommit(tx) } } // BeforeRollback func (mi *mockTxHook) BeforeRollback(tx *Tx) { if mi.beforeRollback != nil { mi.beforeRollback(tx) } } func baseMockConn(mockConn *mock.MockTestDriverConn) { branchStatusCache = gcache.New(1024).LRU().Expiration(time.Minute * 10).Build() mockConn.EXPECT().ExecContext(gomock.Any(), gomock.Any(), gomock.Any()).AnyTimes().Return(&driver.ResultNoRows, nil) mockConn.EXPECT().Exec(gomock.Any(), gomock.Any()).AnyTimes().Return(&driver.ResultNoRows, nil) mockConn.EXPECT().ResetSession(gomock.Any()).AnyTimes().Return(nil) mockConn.EXPECT().Close().AnyTimes().Return(nil) mockConn.EXPECT().QueryContext(gomock.Any(), gomock.Any(), gomock.Any()).AnyTimes().DoAndReturn( func(ctx context.Context, query string, args []driver.NamedValue) (driver.Rows, error) { rows := &mysqlMockRows{} rows.data = [][]interface{}{ {"8.0.29"}, } return rows, nil }) } func initXAConnTestResource(t *testing.T) (*gomock.Controller, *sql.DB, *mockSQLInterceptor, *mockTxHook) { ctrl := gomock.NewController(t) mockMgr := initMockResourceManager(branch.BranchTypeXA, ctrl) _ = mockMgr //db, err := sql.Open("seata-xa-mysql", "root:seata_go@tcp(127.0.0.1:3306)/seata_go_test?multiStatements=true") db, err := sql.Open("seata-xa-mysql", "root:12345678@tcp(127.0.0.1:3306)/seata_client?multiStatements=true&interpolateParams=true") if err != nil { t.Fatal(err) } _ = initMockXaConnector(t, ctrl, db, func(t *testing.T, ctrl *gomock.Controller) driver.Connector { mockTx := mock.NewMockTestDriverTx(ctrl) mockTx.EXPECT().Commit().AnyTimes().Return(nil) mockTx.EXPECT().Rollback().AnyTimes().Return(nil) mockConn := mock.NewMockTestDriverConn(ctrl) mockConn.EXPECT().Begin().AnyTimes().Return(mockTx, nil) mockConn.EXPECT().BeginTx(gomock.Any(), gomock.Any()).AnyTimes().Return(mockTx, nil) baseMockConn(mockConn) connector := mock.NewMockTestDriverConnector(ctrl) connector.EXPECT().Connect(gomock.Any()).AnyTimes().Return(mockConn, nil) return connector }) mi := &mockSQLInterceptor{} ti := &mockTxHook{} exec.CleanCommonHook() CleanTxHooks() exec.RegisterCommonHook(mi) RegisterTxHook(ti) return ctrl, db, mi, ti } func TestXAConn_ExecContext(t *testing.T) { ctrl, db, mi, ti := initXAConnTestResource(t) defer func() { ctrl.Finish() db.Close() CleanTxHooks() }() t.Run("have xid", func(t *testing.T) { ctx := tm.InitSeataContext(context.Background()) tm.SetXID(ctx, uuid.New().String()) before := func(_ context.Context, execCtx *types.ExecContext) { t.Logf("on exec xid=%s", execCtx.TxCtx.XID) assert.Equal(t, tm.GetXID(ctx), execCtx.TxCtx.XID) assert.Equal(t, types.XAMode, execCtx.TxCtx.TransactionMode) } mi.before = before var comitCnt int32 beforeCommit := func(tx *Tx) { atomic.AddInt32(&comitCnt, 1) assert.Equal(t, tx.tranCtx.TransactionMode, types.XAMode) } ti.beforeCommit = beforeCommit conn, err := db.Conn(context.Background()) assert.NoError(t, err) _, err = conn.ExecContext(ctx, "SELECT 1") assert.NoError(t, err) _, err = db.ExecContext(ctx, "SELECT 1") assert.NoError(t, err) // todo fix assert.Equal(t, int32(0), atomic.LoadInt32(&comitCnt)) }) t.Run("not xid", func(t *testing.T) { before := func(_ context.Context, execCtx *types.ExecContext) { assert.Equal(t, "", execCtx.TxCtx.XID) assert.Equal(t, types.Local, execCtx.TxCtx.TransactionMode) } mi.before = before var comitCnt int32 beforeCommit := func(tx *Tx) { atomic.AddInt32(&comitCnt, 1) } ti.beforeCommit = beforeCommit conn, err := db.Conn(context.Background()) assert.NoError(t, err) _, err = conn.ExecContext(context.Background(), "SELECT 1") assert.NoError(t, err) _, err = db.ExecContext(context.Background(), "SELECT 1") assert.NoError(t, err) _, err = db.Exec("SELECT 1") assert.NoError(t, err) assert.Equal(t, int32(0), atomic.LoadInt32(&comitCnt)) }) } func TestXAConn_BeginTx(t *testing.T) { ctrl, db, mi, ti := initXAConnTestResource(t) defer func() { CleanTxHooks() db.Close() ctrl.Finish() }() t.Run("tx-local", func(t *testing.T) { tx, err := db.Begin() assert.NoError(t, err) mi.before = func(_ context.Context, execCtx *types.ExecContext) { assert.Equal(t, "", execCtx.TxCtx.XID) assert.Equal(t, types.Local, execCtx.TxCtx.TransactionMode) } var comitCnt int32 ti.beforeCommit = func(tx *Tx) { atomic.AddInt32(&comitCnt, 1) } _, err = tx.ExecContext(context.Background(), "SELECT * FROM user") assert.NoError(t, err) _, err = tx.ExecContext(tm.InitSeataContext(context.Background()), "SELECT * FROM user") assert.NoError(t, err) err = tx.Commit() assert.NoError(t, err) assert.Equal(t, int32(1), atomic.LoadInt32(&comitCnt)) }) t.Run("tx-local-context", func(t *testing.T) { tx, err := db.BeginTx(context.Background(), &sql.TxOptions{}) assert.NoError(t, err) mi.before = func(_ context.Context, execCtx *types.ExecContext) { assert.Equal(t, "", execCtx.TxCtx.XID) assert.Equal(t, types.Local, execCtx.TxCtx.TransactionMode) } var comitCnt int32 ti.beforeCommit = func(tx *Tx) { atomic.AddInt32(&comitCnt, 1) } _, err = tx.ExecContext(context.Background(), "SELECT * FROM user") assert.NoError(t, err) _, err = tx.ExecContext(tm.InitSeataContext(context.Background()), "SELECT * FROM user") assert.NoError(t, err) err = tx.Commit() assert.NoError(t, err) assert.Equal(t, int32(1), atomic.LoadInt32(&comitCnt)) }) t.Run("tx-xa-context", func(t *testing.T) { ctx := tm.InitSeataContext(context.Background()) tm.SetXID(ctx, uuid.NewString()) tx, err := db.BeginTx(ctx, &sql.TxOptions{}) assert.NoError(t, err) mi.before = func(_ context.Context, execCtx *types.ExecContext) { assert.Equal(t, tm.GetXID(ctx), execCtx.TxCtx.XID) assert.Equal(t, types.XAMode, execCtx.TxCtx.TransactionMode) } var comitCnt int32 ti.beforeCommit = func(tx *Tx) { atomic.AddInt32(&comitCnt, 1) } _, err = tx.ExecContext(context.Background(), "SELECT * FROM user") assert.NoError(t, err) _, err = tx.ExecContext(context.Background(), "SELECT * FROM user") assert.NoError(t, err) err = tx.Commit() assert.NoError(t, err) assert.Equal(t, int32(1), atomic.LoadInt32(&comitCnt)) }) }