diff --git a/meta/autoid/autoid.go b/meta/autoid/autoid.go index f5ad8c78f1fc3..85d5b119c5c3e 100755 --- a/meta/autoid/autoid.go +++ b/meta/autoid/autoid.go @@ -245,6 +245,11 @@ func (alloc *allocator) Rebase(tableID, requiredBase int64, allocIDs bool) error // NextStep return new auto id step according to previous step and consuming time. func NextStep(curStep int64, consumeDur time.Duration) int64 { + failpoint.Inject("mockAutoIDCustomize", func(val failpoint.Value) { + if val.(bool) { + failpoint.Return(3) + } + }) failpoint.Inject("mockAutoIDChange", func(val failpoint.Value) { if val.(bool) { failpoint.Return(step) @@ -387,14 +392,6 @@ func (alloc *allocator) alloc4Signed(tableID int64, n uint64, increment, offset consumeDur := startTime.Sub(alloc.lastAllocTime) nextStep = NextStep(alloc.step, consumeDur) } - // Although the step is customized by user, we still need to make sure nextStep is big enough for insert batch. - if nextStep <= n1 { - nextStep = mathutil.MinInt64(n1*2, maxStep) - } - // Store the step for non-customized-step allocator to calculate next dynamic step. - if !alloc.customStep { - alloc.step = nextStep - } err := kv.RunInNewTxn(alloc.store, true, func(txn kv.Transaction) error { m := meta.NewMeta(txn) var err1 error @@ -402,6 +399,12 @@ func (alloc *allocator) alloc4Signed(tableID int64, n uint64, increment, offset if err1 != nil { return err1 } + // CalcNeededBatchSize calculates the total batch size needed on global base. + n1 = CalcNeededBatchSize(newBase, int64(n), increment, offset, alloc.isUnsigned) + // Although the step is customized by user, we still need to make sure nextStep is big enough for insert batch. + if nextStep < n1 { + nextStep = n1 + } tmpStep := mathutil.MinInt64(math.MaxInt64-newBase, nextStep) // The global rest is not enough for alloc. if tmpStep < n1 { @@ -414,6 +417,10 @@ func (alloc *allocator) alloc4Signed(tableID int64, n uint64, increment, offset if err != nil { return 0, 0, err } + // Store the step for non-customized-step allocator to calculate next dynamic step. + if !alloc.customStep { + alloc.step = nextStep + } alloc.lastAllocTime = time.Now() if newBase == math.MaxInt64 { return 0, 0, ErrAutoincReadFailed @@ -454,14 +461,6 @@ func (alloc *allocator) alloc4Unsigned(tableID int64, n uint64, increment, offse consumeDur := startTime.Sub(alloc.lastAllocTime) nextStep = NextStep(alloc.step, consumeDur) } - // Although the step is customized by user, we still need to make sure nextStep is big enough for insert batch. - if nextStep <= n1 { - nextStep = mathutil.MinInt64(n1*2, maxStep) - } - // Store the step for non-customized-step allocator to calculate next dynamic step. - if !alloc.customStep { - alloc.step = nextStep - } err := kv.RunInNewTxn(alloc.store, true, func(txn kv.Transaction) error { m := meta.NewMeta(txn) var err1 error @@ -469,6 +468,12 @@ func (alloc *allocator) alloc4Unsigned(tableID int64, n uint64, increment, offse if err1 != nil { return err1 } + // CalcNeededBatchSize calculates the total batch size needed on new base. + n1 = CalcNeededBatchSize(newBase, int64(n), increment, offset, alloc.isUnsigned) + // Although the step is customized by user, we still need to make sure nextStep is big enough for insert batch. + if nextStep < n1 { + nextStep = n1 + } tmpStep := int64(mathutil.MinUint64(math.MaxUint64-uint64(newBase), uint64(nextStep))) // The global rest is not enough for alloc. if tmpStep < n1 { @@ -481,6 +486,10 @@ func (alloc *allocator) alloc4Unsigned(tableID int64, n uint64, increment, offse if err != nil { return 0, 0, err } + // Store the step for non-customized-step allocator to calculate next dynamic step. + if !alloc.customStep { + alloc.step = nextStep + } alloc.lastAllocTime = time.Now() if uint64(newBase) == math.MaxUint64 { return 0, 0, ErrAutoincReadFailed @@ -497,3 +506,11 @@ func (alloc *allocator) alloc4Unsigned(tableID int64, n uint64, increment, offse alloc.base = int64(uint64(alloc.base) + uint64(n1)) return min, alloc.base, nil } + +// TestModifyBaseAndEndInjection exported for testing modifying the base and end. +func TestModifyBaseAndEndInjection(alloc Allocator, base, end int64) { + alloc.(*allocator).mu.Lock() + alloc.(*allocator).base = base + alloc.(*allocator).end = end + alloc.(*allocator).mu.Unlock() +} diff --git a/meta/autoid/autoid_test.go b/meta/autoid/autoid_test.go index c1efb8ae711fb..14364d5edba7b 100644 --- a/meta/autoid/autoid_test.go +++ b/meta/autoid/autoid_test.go @@ -565,3 +565,54 @@ func BenchmarkAllocator_Alloc(b *testing.B) { alloc.Alloc(2, 1, 1, 1) } } + +// Fix a computation logic bug in allocator computation. +func (*testSuite) TestAllocComputationIssue(c *C) { + c.Assert(failpoint.Enable("github.com/pingcap/tidb/meta/autoid/mockAutoIDCustomize", `return(true)`), IsNil) + defer func() { + c.Assert(failpoint.Disable("github.com/pingcap/tidb/meta/autoid/mockAutoIDCustomize"), IsNil) + }() + + store, err := mockstore.NewMockTikvStore() + c.Assert(err, IsNil) + defer store.Close() + + err = kv.RunInNewTxn(store, false, func(txn kv.Transaction) error { + m := meta.NewMeta(txn) + err = m.CreateDatabase(&model.DBInfo{ID: 1, Name: model.NewCIStr("a")}) + c.Assert(err, IsNil) + err = m.CreateTableOrView(1, &model.TableInfo{ID: 1, Name: model.NewCIStr("t")}) + c.Assert(err, IsNil) + err = m.CreateTableOrView(1, &model.TableInfo{ID: 2, Name: model.NewCIStr("t1")}) + c.Assert(err, IsNil) + return nil + }) + c.Assert(err, IsNil) + + // Since the test here is applicable to any type of allocators, autoid.RowIDAllocType is chosen. + unsignedAlloc := autoid.NewAllocator(store, 1, true) + c.Assert(unsignedAlloc, NotNil) + signedAlloc := autoid.NewAllocator(store, 1, false) + c.Assert(signedAlloc, NotNil) + + // the next valid two value must be 13 & 16, batch size = 6. + err = unsignedAlloc.Rebase(1, 10, false) + c.Assert(err, IsNil) + // the next valid two value must be 10 & 13, batch size = 6. + err = signedAlloc.Rebase(2, 7, false) + c.Assert(err, IsNil) + // Simulate the rest cache is not enough for next batch, assuming 10 & 13, batch size = 4. + autoid.TestModifyBaseAndEndInjection(unsignedAlloc, 9, 9) + // Simulate the rest cache is not enough for next batch, assuming 10 & 13, batch size = 4. + autoid.TestModifyBaseAndEndInjection(signedAlloc, 4, 6) + + // Here will recompute the new allocator batch size base on new base = 10, which will get 6. + min, max, err := unsignedAlloc.Alloc(1, 2, 3, 1) + c.Assert(err, IsNil) + c.Assert(min, Equals, int64(10)) + c.Assert(max, Equals, int64(16)) + min, max, err = signedAlloc.Alloc(2, 2, 3, 1) + c.Assert(err, IsNil) + c.Assert(min, Equals, int64(7)) + c.Assert(max, Equals, int64(13)) +}