package kv import ( "context" "crypto/rand" "reflect" "testing" "github.com/prysmaticlabs/prysm/shared/bytesutil" bolt "go.etcd.io/bbolt" ) func Test_deleteValueForIndices(t *testing.T) { db := setupDB(t) blocks := make([]byte, 128) if _, err := rand.Read(blocks); err != nil { t.Fatal(err) } tests := []struct { name string inputIndices map[string][]byte root []byte outputIndices map[string][]byte wantErr bool }{ { name: "empty input, no root", inputIndices: map[string][]byte{}, root: []byte{}, outputIndices: map[string][]byte{}, wantErr: false, }, { name: "empty input, root does not exist", inputIndices: map[string][]byte{}, root: bytesutil.PadTo([]byte("not found"), 32), outputIndices: map[string][]byte{}, wantErr: false, }, { name: "non empty input, root does not exist", inputIndices: map[string][]byte{ "blocks": bytesutil.PadTo([]byte{0xde, 0xad, 0xbe, 0xef}, 64), }, root: bytesutil.PadTo([]byte("not found"), 32), outputIndices: map[string][]byte{ "blocks": bytesutil.PadTo([]byte{0xde, 0xad, 0xbe, 0xef}, 64), }, wantErr: false, }, { name: "removes value for a single bucket", inputIndices: map[string][]byte{ "blocks": {0xde, 0xad, 0xbe, 0xef}, }, root: []byte{0xde}, outputIndices: map[string][]byte{ "blocks": {0xad, 0xbe, 0xef}, }, wantErr: false, }, { name: "removes multi-byte value for a single bucket (non-aligned)", inputIndices: map[string][]byte{ "blocks": {0xde, 0xad, 0xbe, 0xef}, }, root: []byte{0xad, 0xbe}, outputIndices: map[string][]byte{ "blocks": {0xde, 0xad, 0xbe, 0xef}, }, wantErr: false, }, { name: "removes multi-byte value for a single bucket (non-aligned)", inputIndices: map[string][]byte{ "blocks": {0xde, 0xad, 0xbe, 0xef, 0xff, 0x01}, }, root: []byte{0xbe, 0xef}, outputIndices: map[string][]byte{ "blocks": {0xde, 0xad, 0xff, 0x01}, }, wantErr: false, }, { name: "removes value from multiple buckets", inputIndices: map[string][]byte{ "blocks": {0xff, 0x32, 0x45, 0x25, 0xde, 0xad, 0xbe, 0xef, 0x24}, "state": {0x01, 0x02, 0x03, 0x04}, "check-point": {0xde, 0xad, 0xbe, 0xef}, "powchain": {0xba, 0xad, 0xb0, 0x00, 0xde, 0xad, 0xbe, 0xef, 0xff}, }, root: []byte{0xde, 0xad, 0xbe, 0xef}, outputIndices: map[string][]byte{ "blocks": {0xff, 0x32, 0x45, 0x25, 0x24}, "state": {0x01, 0x02, 0x03, 0x04}, "check-point": nil, "powchain": {0xba, 0xad, 0xb0, 0x00, 0xff}, }, wantErr: false, }, { name: "root as subsequence of two values (preserve)", inputIndices: map[string][]byte{ "blocks": blocks, }, outputIndices: map[string][]byte{ "blocks": blocks, }, root: blocks[48:80], }, { name: "root as subsequence of two values (remove)", inputIndices: map[string][]byte{ "blocks": blocks, }, outputIndices: map[string][]byte{ "blocks": append(blocks[0:64], blocks[96:]...), }, root: blocks[64:96], }, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { err := db.db.Update(func(tx *bolt.Tx) error { for k, idx := range tt.inputIndices { bkt := tx.Bucket([]byte(k)) if err := bkt.Put(idx, tt.inputIndices[k]); err != nil { t.Fatal(err) } } if err := deleteValueForIndices(context.Background(), tt.inputIndices, tt.root, tx); (err != nil) != tt.wantErr { t.Errorf("deleteValueForIndices() error = %v, wantErr %v", err, tt.wantErr) } // Check updated indices. for k, idx := range tt.inputIndices { bkt := tx.Bucket([]byte(k)) valuesAtIndex := bkt.Get(idx) if !reflect.DeepEqual(valuesAtIndex, tt.outputIndices[k]) { t.Errorf("unexpected output at %q, want: %#v, got: %#v", k, tt.outputIndices[k], valuesAtIndex) } } return nil }) if err != nil { t.Fatal(err) } }) } }