github.com/yinchengtsinghua/golang-Eos-dpos-Ethereum@v0.0.0-20190121132951-92cc4225ed8e/eth/filters/bench_test.go (about) 1 2 //此源码被清华学神尹成大魔王专业翻译分析并修改 3 //尹成QQ77025077 4 //尹成微信18510341407 5 //尹成所在QQ群721929980 6 //尹成邮箱 yinc13@mails.tsinghua.edu.cn 7 //尹成毕业于清华大学,微软区块链领域全球最有价值专家 8 //https://mvp.microsoft.com/zh-cn/PublicProfile/4033620 9 //版权所有2017 Go Ethereum作者 10 //此文件是Go以太坊库的一部分。 11 // 12 //Go-Ethereum库是免费软件:您可以重新分发它和/或修改 13 //根据GNU发布的较低通用公共许可证的条款 14 //自由软件基金会,或者许可证的第3版,或者 15 //(由您选择)任何更高版本。 16 // 17 //Go以太坊图书馆的发行目的是希望它会有用, 18 //但没有任何保证;甚至没有 19 //适销性或特定用途的适用性。见 20 //GNU较低的通用公共许可证,了解更多详细信息。 21 // 22 //你应该收到一份GNU较低级别的公共许可证副本 23 //以及Go以太坊图书馆。如果没有,请参见<http://www.gnu.org/licenses/>。 24 25 package filters 26 27 import ( 28 "bytes" 29 "context" 30 "fmt" 31 "testing" 32 "time" 33 34 "github.com/ethereum/go-ethereum/common" 35 "github.com/ethereum/go-ethereum/common/bitutil" 36 "github.com/ethereum/go-ethereum/core/bloombits" 37 "github.com/ethereum/go-ethereum/core/rawdb" 38 "github.com/ethereum/go-ethereum/core/types" 39 "github.com/ethereum/go-ethereum/ethdb" 40 "github.com/ethereum/go-ethereum/event" 41 "github.com/ethereum/go-ethereum/node" 42 ) 43 44 func BenchmarkBloomBits512(b *testing.B) { 45 benchmarkBloomBits(b, 512) 46 } 47 48 func BenchmarkBloomBits1k(b *testing.B) { 49 benchmarkBloomBits(b, 1024) 50 } 51 52 func BenchmarkBloomBits2k(b *testing.B) { 53 benchmarkBloomBits(b, 2048) 54 } 55 56 func BenchmarkBloomBits4k(b *testing.B) { 57 benchmarkBloomBits(b, 4096) 58 } 59 60 func BenchmarkBloomBits8k(b *testing.B) { 61 benchmarkBloomBits(b, 8192) 62 } 63 64 func BenchmarkBloomBits16k(b *testing.B) { 65 benchmarkBloomBits(b, 16384) 66 } 67 68 func BenchmarkBloomBits32k(b *testing.B) { 69 benchmarkBloomBits(b, 32768) 70 } 71 72 const benchFilterCnt = 2000 73 74 func benchmarkBloomBits(b *testing.B, sectionSize uint64) { 75 benchDataDir := node.DefaultDataDir() + "/geth/chaindata" 76 fmt.Println("Running bloombits benchmark section size:", sectionSize) 77 78 db, err := ethdb.NewLDBDatabase(benchDataDir, 128, 1024) 79 if err != nil { 80 b.Fatalf("error opening database at %v: %v", benchDataDir, err) 81 } 82 head := rawdb.ReadHeadBlockHash(db) 83 if head == (common.Hash{}) { 84 b.Fatalf("chain data not found at %v", benchDataDir) 85 } 86 87 clearBloomBits(db) 88 fmt.Println("Generating bloombits data...") 89 headNum := rawdb.ReadHeaderNumber(db, head) 90 if headNum == nil || *headNum < sectionSize+512 { 91 b.Fatalf("not enough blocks for running a benchmark") 92 } 93 94 start := time.Now() 95 cnt := (*headNum - 512) / sectionSize 96 var dataSize, compSize uint64 97 for sectionIdx := uint64(0); sectionIdx < cnt; sectionIdx++ { 98 bc, err := bloombits.NewGenerator(uint(sectionSize)) 99 if err != nil { 100 b.Fatalf("failed to create generator: %v", err) 101 } 102 var header *types.Header 103 for i := sectionIdx * sectionSize; i < (sectionIdx+1)*sectionSize; i++ { 104 hash := rawdb.ReadCanonicalHash(db, i) 105 header = rawdb.ReadHeader(db, hash, i) 106 if header == nil { 107 b.Fatalf("Error creating bloomBits data") 108 } 109 bc.AddBloom(uint(i-sectionIdx*sectionSize), header.Bloom) 110 } 111 sectionHead := rawdb.ReadCanonicalHash(db, (sectionIdx+1)*sectionSize-1) 112 for i := 0; i < types.BloomBitLength; i++ { 113 data, err := bc.Bitset(uint(i)) 114 if err != nil { 115 b.Fatalf("failed to retrieve bitset: %v", err) 116 } 117 comp := bitutil.CompressBytes(data) 118 dataSize += uint64(len(data)) 119 compSize += uint64(len(comp)) 120 rawdb.WriteBloomBits(db, uint(i), sectionIdx, sectionHead, comp) 121 } 122 //如果节IDX%50==0 123 //fmt.println(“节”,节IDX,“/”,cnt) 124 //} 125 } 126 127 d := time.Since(start) 128 fmt.Println("Finished generating bloombits data") 129 fmt.Println(" ", d, "total ", d/time.Duration(cnt*sectionSize), "per block") 130 fmt.Println(" data size:", dataSize, " compressed size:", compSize, " compression ratio:", float64(compSize)/float64(dataSize)) 131 132 fmt.Println("Running filter benchmarks...") 133 start = time.Now() 134 mux := new(event.TypeMux) 135 var backend *testBackend 136 137 for i := 0; i < benchFilterCnt; i++ { 138 if i%20 == 0 { 139 db.Close() 140 db, _ = ethdb.NewLDBDatabase(benchDataDir, 128, 1024) 141 backend = &testBackend{mux, db, cnt, new(event.Feed), new(event.Feed), new(event.Feed), new(event.Feed)} 142 } 143 var addr common.Address 144 addr[0] = byte(i) 145 addr[1] = byte(i / 256) 146 filter := NewRangeFilter(backend, 0, int64(cnt*sectionSize-1), []common.Address{addr}, nil) 147 if _, err := filter.Logs(context.Background()); err != nil { 148 b.Error("filter.Find error:", err) 149 } 150 } 151 d = time.Since(start) 152 fmt.Println("Finished running filter benchmarks") 153 fmt.Println(" ", d, "total ", d/time.Duration(benchFilterCnt), "per address", d*time.Duration(1000000)/time.Duration(benchFilterCnt*cnt*sectionSize), "per million blocks") 154 db.Close() 155 } 156 157 func forEachKey(db ethdb.Database, startPrefix, endPrefix []byte, fn func(key []byte)) { 158 it := db.(*ethdb.LDBDatabase).NewIterator() 159 it.Seek(startPrefix) 160 for it.Valid() { 161 key := it.Key() 162 cmpLen := len(key) 163 if len(endPrefix) < cmpLen { 164 cmpLen = len(endPrefix) 165 } 166 if bytes.Compare(key[:cmpLen], endPrefix) == 1 { 167 break 168 } 169 fn(common.CopyBytes(key)) 170 it.Next() 171 } 172 it.Release() 173 } 174 175 var bloomBitsPrefix = []byte("bloomBits-") 176 177 func clearBloomBits(db ethdb.Database) { 178 fmt.Println("Clearing bloombits data...") 179 forEachKey(db, bloomBitsPrefix, bloomBitsPrefix, func(key []byte) { 180 db.Delete(key) 181 }) 182 } 183 184 func BenchmarkNoBloomBits(b *testing.B) { 185 benchDataDir := node.DefaultDataDir() + "/geth/chaindata" 186 fmt.Println("Running benchmark without bloombits") 187 db, err := ethdb.NewLDBDatabase(benchDataDir, 128, 1024) 188 if err != nil { 189 b.Fatalf("error opening database at %v: %v", benchDataDir, err) 190 } 191 head := rawdb.ReadHeadBlockHash(db) 192 if head == (common.Hash{}) { 193 b.Fatalf("chain data not found at %v", benchDataDir) 194 } 195 headNum := rawdb.ReadHeaderNumber(db, head) 196 197 clearBloomBits(db) 198 199 fmt.Println("Running filter benchmarks...") 200 start := time.Now() 201 mux := new(event.TypeMux) 202 backend := &testBackend{mux, db, 0, new(event.Feed), new(event.Feed), new(event.Feed), new(event.Feed)} 203 filter := NewRangeFilter(backend, 0, int64(*headNum), []common.Address{{}}, nil) 204 filter.Logs(context.Background()) 205 d := time.Since(start) 206 fmt.Println("Finished running filter benchmarks") 207 fmt.Println(" ", d, "total ", d*time.Duration(1000000)/time.Duration(*headNum+1), "per million blocks") 208 db.Close() 209 }