// Copyright 2025 The go-ethereum Authors // This file is part of the go-ethereum library. // // The go-ethereum library is free software: you can redistribute it and/or modify // it under the terms of the GNU Lesser General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // The go-ethereum library is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public License // along with the go-ethereum library. If not, see value }) got, err := br.readGreaterThan(value) if err != nil { t.Fatalf("Unexpected error, got %v", err) } if pos == len(elements) { if got != math.MaxUint64 { t.Fatalf("Unexpected result, got %d, wanted math.MaxUint64", got) } } else if got != elements[pos] { t.Fatalf("Unexpected result, got %d, wanted %d", got, elements[pos]) } } } func TestEmptyIndexReader(t *testing.T) { br, err := newIndexReader(rawdb.NewMemoryDatabase(), newAccountIdent(common.Address{0xa})) if err != nil { t.Fatalf("Failed to construct the index reader, %v", err) } res, err := br.readGreaterThan(100) if err != nil { t.Fatalf("Failed to query, %v", err) } if res != math.MaxUint64 { t.Fatalf("Unexpected result, got %d, wanted math.MaxUint64", res) } } func TestIndexWriterBasic(t *testing.T) { db := rawdb.NewMemoryDatabase() iw, _ := newIndexWriter(db, newAccountIdent(common.Address{0xa})) iw.append(2) if err := iw.append(1); err == nil { t.Fatal("out-of-order insertion is not expected") } for i := 0; i < 10; i++ { iw.append(uint64(i + 3)) } batch := db.NewBatch() iw.finish(batch) batch.Write() iw, err := newIndexWriter(db, newAccountIdent(common.Address{0xa})) if err != nil { t.Fatalf("Failed to construct the block writer, %v", err) } for i := 0; i < 10; i++ { if err := iw.append(uint64(i + 100)); err != nil { t.Fatalf("Failed to append item, %v", err) } } iw.finish(db.NewBatch()) } func TestIndexWriterDelete(t *testing.T) { db := rawdb.NewMemoryDatabase() iw, _ := newIndexWriter(db, newAccountIdent(common.Address{0xa})) for i := 0; i < indexBlockEntriesCap*4; i++ { iw.append(uint64(i + 1)) } batch := db.NewBatch() iw.finish(batch) batch.Write() // Delete unknown id, the request should be rejected id, _ := newIndexDeleter(db, newAccountIdent(common.Address{0xa})) if err := id.pop(indexBlockEntriesCap * 5); err == nil { t.Fatal("Expect error to occur for unknown id") } for i := indexBlockEntriesCap * 4; i >= 1; i-- { if err := id.pop(uint64(i)); err != nil { t.Fatalf("Unexpected error for element popping, %v", err) } if id.lastID != uint64(i-1) { t.Fatalf("Unexpected lastID, want: %d, got: %d", uint64(i-1), iw.lastID) } if rand.Intn(10) == 0 { batch := db.NewBatch() id.finish(batch) batch.Write() } } } func TestBatchIndexerWrite(t *testing.T) { var ( db = rawdb.NewMemoryDatabase() batch = newBatchIndexer(db, false) histories = makeHistories(10) ) for i, h := range histories { if err := batch.process(h, uint64(i+1)); err != nil { t.Fatalf("Failed to process history, %v", err) } } if err := batch.finish(true); err != nil { t.Fatalf("Failed to finish batch indexer, %v", err) } indexed := rawdb.ReadLastStateHistoryIndex(db) if indexed == nil || *indexed != uint64(10) { t.Fatal("Unexpected index position") } var ( accounts = make(map[common.Address][]uint64) storages = make(map[common.Address]map[common.Hash][]uint64) ) for i, h := range histories { for _, addr := range h.accountList { accounts[addr] = append(accounts[addr], uint64(i+1)) if _, ok := storages[addr]; !ok { storages[addr] = make(map[common.Hash][]uint64) } for _, slot := range h.storageList[addr] { storages[addr][slot] = append(storages[addr][slot], uint64(i+1)) } } } for addr, indexes := range accounts { ir, _ := newIndexReader(db, newAccountIdent(addr)) for i := 0; i < len(indexes)-1; i++ { n, err := ir.readGreaterThan(indexes[i]) if err != nil { t.Fatalf("Failed to read index, %v", err) } if n != indexes[i+1] { t.Fatalf("Unexpected result, want %d, got %d", indexes[i+1], n) } } n, err := ir.readGreaterThan(indexes[len(indexes)-1]) if err != nil { t.Fatalf("Failed to read index, %v", err) } if n != math.MaxUint64 { t.Fatalf("Unexpected result, want math.MaxUint64, got %d", n) } } for addr, slots := range storages { for slotHash, indexes := range slots { ir, _ := newIndexReader(db, newStorageIdent(addr, slotHash)) for i := 0; i < len(indexes)-1; i++ { n, err := ir.readGreaterThan(indexes[i]) if err != nil { t.Fatalf("Failed to read index, %v", err) } if n != indexes[i+1] { t.Fatalf("Unexpected result, want %d, got %d", indexes[i+1], n) } } n, err := ir.readGreaterThan(indexes[len(indexes)-1]) if err != nil { t.Fatalf("Failed to read index, %v", err) } if n != math.MaxUint64 { t.Fatalf("Unexpected result, want math.MaxUint64, got %d", n) } } } } func TestBatchIndexerDelete(t *testing.T) { var ( db = rawdb.NewMemoryDatabase() bw = newBatchIndexer(db, false) histories = makeHistories(10) ) // Index histories for i, h := range histories { if err := bw.process(h, uint64(i+1)); err != nil { t.Fatalf("Failed to process history, %v", err) } } if err := bw.finish(true); err != nil { t.Fatalf("Failed to finish batch indexer, %v", err) } // Unindex histories bd := newBatchIndexer(db, true) for i := len(histories) - 1; i >= 0; i-- { if err := bd.process(histories[i], uint64(i+1)); err != nil { t.Fatalf("Failed to process history, %v", err) } } if err := bd.finish(true); err != nil { t.Fatalf("Failed to finish batch indexer, %v", err) } indexed := rawdb.ReadLastStateHistoryIndex(db) if indexed != nil { t.Fatal("Unexpected index position") } it := db.NewIterator(rawdb.StateHistoryIndexPrefix, nil) for it.Next() { t.Fatal("Leftover history index data") } it.Release() }