mirror of https://github.com/tikv/client-go.git
148 lines
5.0 KiB
Go
148 lines
5.0 KiB
Go
// Copyright 2021 TiKV Authors
|
|
//
|
|
// Licensed 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.
|
|
|
|
// NOTE: The code in this file is based on code from the
|
|
// TiDB project, licensed under the Apache License v 2.0
|
|
//
|
|
// https://github.com/pingcap/tidb/tree/cc5e161ac06827589c4966674597c137cc9e809c/store/tikv/tests/prewrite_test.go
|
|
//
|
|
|
|
// Copyright 2020 PingCAP, Inc.
|
|
//
|
|
// Licensed 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 tikv_test
|
|
|
|
import (
|
|
"context"
|
|
"sync"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/pingcap/failpoint"
|
|
"github.com/pingcap/kvproto/pkg/kvrpcpb"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
"github.com/tikv/client-go/v2/testutils"
|
|
"github.com/tikv/client-go/v2/tikv"
|
|
"github.com/tikv/client-go/v2/tikvrpc"
|
|
"github.com/tikv/client-go/v2/txnkv/transaction"
|
|
)
|
|
|
|
func TestSetMinCommitTSInAsyncCommit(t *testing.T) {
|
|
require, assert := require.New(t), assert.New(t)
|
|
store := NewTestStore(t)
|
|
defer store.Close()
|
|
|
|
tx, err := store.Begin()
|
|
require.Nil(err)
|
|
txn := transaction.TxnProbe{KVTxn: tx}
|
|
err = txn.Set([]byte("k"), []byte("v"))
|
|
assert.Nil(err)
|
|
committer, err := txn.NewCommitter(1)
|
|
assert.Nil(err)
|
|
committer.SetUseAsyncCommit()
|
|
|
|
buildRequest := func() *kvrpcpb.PrewriteRequest {
|
|
req := committer.BuildPrewriteRequest(1, 1, 1, committer.GetMutations(), 1)
|
|
return req.Req.(*kvrpcpb.PrewriteRequest)
|
|
}
|
|
|
|
// no forUpdateTS
|
|
req := buildRequest()
|
|
assert.Equal(req.MinCommitTs, txn.StartTS()+1)
|
|
|
|
// forUpdateTS is set
|
|
committer.SetForUpdateTS(txn.StartTS() + (5 << 18))
|
|
req = buildRequest()
|
|
assert.Equal(req.MinCommitTs, committer.GetForUpdateTS()+1)
|
|
|
|
// minCommitTS is set
|
|
committer.SetMinCommitTS(txn.StartTS() + (10 << 18))
|
|
req = buildRequest()
|
|
assert.Equal(req.MinCommitTs, committer.GetMinCommitTS())
|
|
|
|
}
|
|
|
|
// TestIsRetryRequestFlagWithRegionError tests that the is_retry_request flag is true for all retrying prewrite requests.
|
|
func TestIsRetryRequestFlagWithRegionError(t *testing.T) {
|
|
require := require.New(t)
|
|
|
|
client, cluster, pdClient, err := testutils.NewMockTiKV("", nil)
|
|
require.Nil(err)
|
|
_, peerID, regionID := testutils.BootstrapWithSingleStore(cluster)
|
|
store, err := tikv.NewTestTiKVStore(client, pdClient, nil, nil, 0)
|
|
require.Nil(err)
|
|
defer store.Close()
|
|
|
|
failpoint.Enable("tikvclient/mockRetrySendReqToRegion", "1*return(true)->return(false)")
|
|
defer failpoint.Disable("tikvclient/mockRetrySendReqToRegion")
|
|
failpoint.Enable("tikvclient/invalidCacheAndRetry", "1*off->pause")
|
|
|
|
// the history of this field
|
|
isRetryRequest := make([]bool, 0, 0)
|
|
var mu sync.Mutex
|
|
hook := func(req *tikvrpc.Request) {
|
|
if req.Type != tikvrpc.CmdPrewrite {
|
|
return
|
|
}
|
|
if req != nil {
|
|
mu.Lock()
|
|
isRetryRequest = append(isRetryRequest, req.Context.IsRetryRequest)
|
|
mu.Unlock()
|
|
}
|
|
}
|
|
failpoint.Enable("tikvclient/beforeSendReqToRegion", "return")
|
|
defer failpoint.Disable("tikvclient/beforeSendReqToRegion")
|
|
ctx := context.WithValue(context.TODO(), "sendReqToRegionHook", hook)
|
|
|
|
tx, err := store.Begin()
|
|
require.Nil(err)
|
|
txn := transaction.TxnProbe{KVTxn: tx}
|
|
err = txn.Set([]byte("a"), []byte("v"))
|
|
require.Nil(err)
|
|
err = txn.Set([]byte("z"), []byte("v"))
|
|
require.Nil(err)
|
|
committer, err := txn.NewCommitter(1)
|
|
require.Nil(err)
|
|
var wg sync.WaitGroup
|
|
wg.Add(1)
|
|
go func() {
|
|
committer.PrewriteAllMutations(ctx)
|
|
wg.Done()
|
|
}()
|
|
time.Sleep(time.Second * 3)
|
|
cluster.Split(regionID, cluster.AllocID(), []byte("h"), []uint64{peerID}, peerID)
|
|
failpoint.Disable("tikvclient/invalidCacheAndRetry")
|
|
wg.Wait()
|
|
|
|
// The event history should be:
|
|
// 1. The first prewrite succeeds in TiKV, but due to some reason, client-go doesn't get the response. We inject a retry to simulate it.
|
|
// 2. The second prewrite request returns a region error, which is caused by the region split. And it retries.
|
|
// 3. The third and fourth prewrite requests (for 'a' and 'z' respectively, so there are 2 of them) succeed.
|
|
//
|
|
// The last three requests are retry requests, we assert the is_retry_request flags are all true.
|
|
require.Equal([]bool{false, true, true, true}, isRetryRequest)
|
|
}
|