Files
tidb/cmd/ddltest/column_test.go

209 lines
6.1 KiB
Go

// Copyright 2015 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 ddltest
import (
"fmt"
"reflect"
"sync"
"sync/atomic"
"testing"
"time"
"github.com/pingcap/log"
"github.com/pingcap/tidb/kv"
"github.com/pingcap/tidb/table"
"github.com/pingcap/tidb/table/tables"
"github.com/pingcap/tidb/types"
"github.com/stretchr/testify/require"
"go.uber.org/zap"
goctx "golang.org/x/net/context"
)
// After add column finished, check the records in the table.
func (s *ddlSuite) checkAddColumn(t *testing.T, rowID int64, defaultVal interface{}, updatedVal interface{}) {
ctx := s.ctx
err := ctx.NewTxn(goctx.Background())
require.NoError(t, err)
tbl := s.getTable(t, "test_column")
oldInsertCount := int64(0)
newInsertCount := int64(0)
oldUpdateCount := int64(0)
newUpdateCount := int64(0)
err = tables.IterRecords(tbl, ctx, tbl.Cols(), func(_ kv.Handle, data []types.Datum, cols []*table.Column) (bool, error) {
col1Val := data[0].GetValue()
col2Val := data[1].GetValue()
col3Val := data[2].GetValue()
// Check inserted row.
if reflect.DeepEqual(col1Val, col2Val) {
if reflect.DeepEqual(col3Val, defaultVal) {
// When insert a row with 2 columns, the third column will be default value.
oldInsertCount++
} else if reflect.DeepEqual(col3Val, col1Val) {
// When insert a row with 3 columns, the third column value will be the first column value.
newInsertCount++
} else {
log.Fatal("[checkAddColumn fail]invalid row", zap.Any("row", data))
}
}
// Check updated row.
if reflect.DeepEqual(col2Val, updatedVal) {
if reflect.DeepEqual(col3Val, defaultVal) || reflect.DeepEqual(col3Val, col1Val) {
oldUpdateCount++
} else if reflect.DeepEqual(col3Val, updatedVal) {
newUpdateCount++
} else {
log.Fatal("[checkAddColumn fail]invalid row", zap.Any("row", data))
}
}
return true, nil
})
require.NoError(t, err)
deleteCount := rowID - oldInsertCount - newInsertCount - oldUpdateCount - newUpdateCount
require.GreaterOrEqual(t, oldInsertCount, int64(0))
require.GreaterOrEqual(t, newInsertCount, int64(0))
require.Greater(t, oldUpdateCount, int64(0))
require.Greater(t, newUpdateCount, int64(0))
require.Greater(t, deleteCount, int64(0))
}
func (s *ddlSuite) checkDropColumn(t *testing.T, rowID int64, alterColumn *table.Column, updateDefault interface{}) {
ctx := s.ctx
err := ctx.NewTxn(goctx.Background())
require.NoError(t, err)
tbl := s.getTable(t, "test_column")
for _, col := range tbl.Cols() {
require.NotEqual(t, alterColumn.ID, col.ID)
}
insertCount := int64(0)
updateCount := int64(0)
err = tables.IterRecords(tbl, ctx, tbl.Cols(), func(_ kv.Handle, data []types.Datum, cols []*table.Column) (bool, error) {
if reflect.DeepEqual(data[1].GetValue(), data[0].GetValue()) {
// Check inserted row.
insertCount++
} else if reflect.DeepEqual(data[1].GetValue(), updateDefault) {
// Check updated row.
updateCount++
} else {
log.Fatal("[checkDropColumn fail]invalid row", zap.Any("row", data))
}
return true, nil
})
require.NoError(t, err)
deleteCount := rowID - insertCount - updateCount
require.Greater(t, insertCount, int64(0))
require.Greater(t, updateCount, int64(0))
require.Greater(t, deleteCount, int64(0))
}
func TestColumn(t *testing.T) {
s := createDDLSuite(t)
defer s.teardown(t)
// first add many data
workerNum := 10
base := *dataNum / workerNum
var wg sync.WaitGroup
wg.Add(workerNum)
for i := 0; i < workerNum; i++ {
go func(i int) {
defer wg.Done()
for j := 0; j < base; j++ {
k := base*i + j
s.execInsert(fmt.Sprintf("insert into test_column values (%d, %d)", k, k))
}
}(i)
}
wg.Wait()
tbl := []struct {
Query string
ColumnName string
Add bool
Default interface{}
}{
{"alter table test_column add column c3 int default -1", "c3", true, int64(-1)},
{"alter table test_column drop column c3", "c3", false, nil},
}
rowID := int64(*dataNum)
updateDefault := int64(-2)
var alterColumn *table.Column
for _, col := range tbl {
t.Run(col.Query, func(t *testing.T) {
done := s.runDDL(col.Query)
ticker := time.NewTicker(time.Duration(*lease) * time.Second / 2)
defer ticker.Stop()
LOOP:
for {
select {
case err := <-done:
require.NoError(t, err)
break LOOP
case <-ticker.C:
count := 10
s.execColumnOperations(t, workerNum, count, &rowID, updateDefault)
}
}
if col.Add {
s.checkAddColumn(t, rowID, col.Default, updateDefault)
} else {
s.checkDropColumn(t, rowID, alterColumn, updateDefault)
}
tbl := s.getTable(t, "test_column")
alterColumn = table.FindCol(tbl.Cols(), col.ColumnName)
if col.Add {
require.NotNil(t, alterColumn)
} else {
require.Nil(t, alterColumn)
}
})
}
}
func (s *ddlSuite) execColumnOperations(t *testing.T, workerNum, count int, rowID *int64, updateDefault int64) {
var wg sync.WaitGroup
// workerNum = 10
wg.Add(workerNum)
for i := 0; i < workerNum; i++ {
go func() {
defer wg.Done()
for j := 0; j < count; j++ {
key := int(atomic.AddInt64(rowID, 2))
s.execInsert(fmt.Sprintf("insert into test_column (c1, c2) values (%d, %d)",
key-1, key-1))
_, _ = s.exec(fmt.Sprintf("insert into test_column values (%d, %d, %d)", key, key, key))
s.mustExec(fmt.Sprintf("update test_column set c2 = %d where c1 = %d",
updateDefault, randomNum(key)))
_, _ = s.exec(fmt.Sprintf("update test_column set c2 = %d, c3 = %d where c1 = %d",
updateDefault, updateDefault, randomNum(key)))
s.mustExec(fmt.Sprintf("delete from test_column where c1 = %d", randomNum(key)))
}
}()
}
wg.Wait()
}