// 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 structure_test import ( "context" "testing" "github.com/pingcap/tidb/pkg/kv" "github.com/pingcap/tidb/pkg/parser/mysql" "github.com/pingcap/tidb/pkg/parser/terror" "github.com/pingcap/tidb/pkg/structure" "github.com/pingcap/tidb/pkg/testkit" "github.com/stretchr/testify/require" ) func TestString(t *testing.T) { store := testkit.CreateMockStore(t) txn, err := store.Begin() require.NoError(t, err) tx := structure.NewStructure(txn, txn, []byte{0x00}) key := []byte("a") value := []byte("1") err = tx.Set(key, value) require.NoError(t, err) tx.Iterate([]byte("a"), []byte("b"), func(k, v []byte) error { require.Equal(t, k, key) require.Equal(t, v, value) return nil }) v, err := tx.Get(key) require.NoError(t, err) require.Equal(t, value, v) n, err := tx.Inc(key, 1) require.NoError(t, err) require.Equal(t, int64(2), n) v, err = tx.Get(key) require.NoError(t, err) require.Equal(t, []byte("2"), v) n, err = tx.GetInt64(key) require.NoError(t, err) require.Equal(t, int64(2), n) err = tx.Clear(key) require.NoError(t, err) v, err = tx.Get(key) require.NoError(t, err) require.Nil(t, v) tx1 := structure.NewStructure(txn, nil, []byte{0x01}) err = tx1.Set(key, value) require.NotNil(t, err) _, err = tx1.Inc(key, 1) require.NotNil(t, err) err = tx1.Clear(key) require.NotNil(t, err) err = txn.Commit(context.Background()) require.NoError(t, err) } func TestList(t *testing.T) { store := testkit.CreateMockStore(t) txn, err := store.Begin() require.NoError(t, err) tx := structure.NewStructure(txn, txn, []byte{0x00}) key := []byte("a") err = tx.LPush(key, []byte("3"), []byte("2"), []byte("1")) require.NoError(t, err) // Test LGetAll. err = tx.LPush(key, []byte("11")) require.NoError(t, err) values, err := tx.LGetAll(key) require.NoError(t, err) require.Equal(t, [][]byte{[]byte("3"), []byte("2"), []byte("1"), []byte("11")}, values) value, err := tx.LPop(key) require.NoError(t, err) require.Equal(t, []byte("11"), value) l, err := tx.LLen(key) require.NoError(t, err) require.Equal(t, int64(3), l) value, err = tx.LIndex(key, 1) require.NoError(t, err) require.Equal(t, []byte("2"), value) err = tx.LSet(key, 1, []byte("4")) require.NoError(t, err) value, err = tx.LIndex(key, 1) require.NoError(t, err) require.Equal(t, []byte("4"), value) err = tx.LSet(key, 1, []byte("2")) require.NoError(t, err) err = tx.LSet(key, 100, []byte("2")) require.NotNil(t, err) value, err = tx.LIndex(key, -1) require.NoError(t, err) require.Equal(t, []byte("3"), value) value, err = tx.LPop(key) require.NoError(t, err) require.Equal(t, []byte("1"), value) l, err = tx.LLen(key) require.NoError(t, err) require.Equal(t, int64(2), l) err = tx.RPush(key, []byte("4")) require.NoError(t, err) l, err = tx.LLen(key) require.NoError(t, err) require.Equal(t, int64(3), l) value, err = tx.LIndex(key, -1) require.NoError(t, err) require.Equal(t, []byte("4"), value) value, err = tx.RPop(key) require.NoError(t, err) require.Equal(t, []byte("4"), value) value, err = tx.RPop(key) require.NoError(t, err) require.Equal(t, []byte("3"), value) value, err = tx.RPop(key) require.NoError(t, err) require.Equal(t, []byte("2"), value) l, err = tx.LLen(key) require.NoError(t, err) require.Equal(t, int64(0), l) err = tx.LPush(key, []byte("1")) require.NoError(t, err) err = tx.LClear(key) require.NoError(t, err) l, err = tx.LLen(key) require.NoError(t, err) require.Equal(t, int64(0), l) tx1 := structure.NewStructure(txn, nil, []byte{0x01}) err = tx1.LPush(key, []byte("1")) require.NotNil(t, err) require.NotNil(t, err) _, err = tx1.RPop(key) require.NotNil(t, err) err = tx1.LSet(key, 1, []byte("2")) require.NotNil(t, err) err = tx1.LClear(key) require.NotNil(t, err) err = txn.Commit(context.Background()) require.NoError(t, err) } func TestHash(t *testing.T) { store := testkit.CreateMockStore(t) txn, err := store.Begin() require.NoError(t, err) tx := structure.NewStructure(txn, txn, []byte{0x00}) key := []byte("a") tx.EncodeHashAutoIDKeyValue(key, key, 5) err = tx.HSet(key, []byte("1"), []byte("1")) require.NoError(t, err) err = tx.HSet(key, []byte("2"), []byte("2")) require.NoError(t, err) value, err := tx.HGet(key, []byte("1")) require.NoError(t, err) require.Equal(t, []byte("1"), value) value, err = tx.HGet(key, []byte("fake")) require.NoError(t, err) require.Nil(t, value) keys, err := tx.HKeys(key) require.NoError(t, err) require.Equal(t, [][]byte{[]byte("1"), []byte("2")}, keys) res, err := tx.HGetAll(key) require.NoError(t, err) require.Equal(t, []structure.HashPair{ {Field: []byte("1"), Value: []byte("1")}, {Field: []byte("2"), Value: []byte("2")}}, res) idx := 0 err = tx.HGetIter(key, func(pair structure.HashPair) error { require.Less(t, idx, 2) require.Equal(t, res[idx], pair) idx += 1 return nil }) require.NoError(t, err) res, err = tx.HGetLastN(key, 1) require.NoError(t, err) require.Equal(t, []structure.HashPair{ {Field: []byte("2"), Value: []byte("2")}}, res) res, err = tx.HGetLastN(key, 2) require.NoError(t, err) require.Equal(t, []structure.HashPair{ {Field: []byte("2"), Value: []byte("2")}, {Field: []byte("1"), Value: []byte("1")}}, res) err = tx.HDel(key, []byte("1")) require.NoError(t, err) value, err = tx.HGet(key, []byte("1")) require.NoError(t, err) require.Nil(t, value) n, err := tx.HInc(key, []byte("1"), 1) require.NoError(t, err) require.Equal(t, int64(1), n) // Test set new value which equals to old value. value, err = tx.HGet(key, []byte("1")) require.NoError(t, err) require.Equal(t, []byte("1"), value) err = tx.HSet(key, []byte("1"), []byte("1")) require.NoError(t, err) value, err = tx.HGet(key, []byte("1")) require.NoError(t, err) require.Equal(t, []byte("1"), value) n, err = tx.HInc(key, []byte("1"), 1) require.NoError(t, err) require.Equal(t, int64(2), n) n, err = tx.HInc(key, []byte("1"), 1) require.NoError(t, err) require.Equal(t, int64(3), n) n, err = tx.HGetInt64(key, []byte("1")) require.NoError(t, err) require.Equal(t, int64(3), n) err = tx.HClear(key) require.NoError(t, err) err = tx.HDel(key, []byte("fake_key")) require.NoError(t, err) // Test set nil value. value, err = tx.HGet(key, []byte("nil_key")) require.NoError(t, err) require.Nil(t, value) err = tx.HSet(key, []byte("nil_key"), nil) require.NoError(t, err) err = tx.HSet(key, []byte("nil_key"), []byte("1")) require.NoError(t, err) value, err = tx.HGet(key, []byte("nil_key")) require.NoError(t, err) require.Equal(t, []byte("1"), value) err = tx.HSet(key, []byte("nil_key"), nil) require.NotNil(t, err) value, err = tx.HGet(key, []byte("nil_key")) require.NoError(t, err) require.Equal(t, []byte("1"), value) err = tx.HSet(key, []byte("nil_key"), []byte("2")) require.NoError(t, err) value, err = tx.HGet(key, []byte("nil_key")) require.NoError(t, err) require.Equal(t, []byte("2"), value) tx1 := structure.NewStructure(txn, nil, []byte{0x01}) _, err = tx1.HInc(key, []byte("1"), 1) require.NotNil(t, err) err = tx1.HDel(key, []byte("1")) require.NotNil(t, err) err = txn.Commit(context.Background()) require.NoError(t, err) ctx := kv.WithInternalSourceType(context.Background(), kv.InternalTxnMeta) err = kv.RunInNewTxn(ctx, store, false, func(ctx context.Context, txn kv.Transaction) error { newTxn := structure.NewStructure(txn, txn, []byte{0x00}) err = newTxn.Set(key, []byte("abc")) require.NoError(t, err) value, err = newTxn.Get(key) require.NoError(t, err) require.Equal(t, []byte("abc"), value) return nil }) require.NoError(t, err) } func TestError(t *testing.T) { kvErrs := []*terror.Error{ structure.ErrInvalidHashKeyFlag, structure.ErrInvalidListIndex, structure.ErrInvalidListMetaData, structure.ErrWriteOnSnapshot, } for _, err := range kvErrs { code := terror.ToSQLError(err).Code require.NotEqual(t, mysql.ErrUnknown, code) require.Equal(t, uint16(err.Code()), code, "err: %v", err) } } func TestIterateHashWithBoundedKey(t *testing.T) { store := testkit.CreateMockStore(t) txn, err := store.Begin() require.NoError(t, err) tx := structure.NewStructure(txn, txn, nil) require.NoError(t, tx.Set(tx.EncodeHashMetaKey([]byte("aa")), []byte("meta"))) require.NoError(t, tx.HSet([]byte("b"), []byte(""), []byte("value1"))) require.NoError(t, tx.Set(tx.EncodeHashMetaKey([]byte("c")), []byte("meta"))) require.NoError(t, tx.HSet([]byte("d"), []byte(""), []byte("value2"))) require.NoError(t, tx.Set(tx.EncodeHashMetaKey([]byte("dd")), []byte("meta"))) cnt := 0 require.NoError(t, tx.IterateHashWithBoundedKey([]byte("a"), []byte("e"), func(key, field, value []byte) error { cnt++ return nil })) require.Equal(t, 2, cnt) }